mirror of
https://github.com/unicode-org/icu.git
synced 2025-04-05 21:45:37 +00:00
ICU-5454 Merging ICU4C RuleBasedTimeZone/TimeZoneRule/VTimeZone implementation into trunk
X-SVN-Rev: 21845
This commit is contained in:
parent
957ecb06dd
commit
b340ff12f6
26 changed files with 9016 additions and 37 deletions
14
.gitattributes
vendored
14
.gitattributes
vendored
|
@ -123,10 +123,22 @@ icu4c/source/data/xml/main/nr.xml -text
|
|||
icu4c/source/data/xml/main/pa_Guru.xml -text
|
||||
icu4c/source/data/xml/main/tl.xml -text
|
||||
icu4c/source/data/xml/main/to.xml -text
|
||||
icu4c/source/i18n/basictz.cpp -text
|
||||
icu4c/source/i18n/dtrule.cpp -text
|
||||
icu4c/source/i18n/persncal.cpp -text
|
||||
icu4c/source/i18n/persncal.h -text
|
||||
icu4c/source/i18n/rbtz.cpp -text
|
||||
icu4c/source/i18n/reldtfmt.cpp -text
|
||||
icu4c/source/i18n/reldtfmt.h -text
|
||||
icu4c/source/i18n/tzrule.cpp -text
|
||||
icu4c/source/i18n/tztrans.cpp -text
|
||||
icu4c/source/i18n/unicode/basictz.h -text
|
||||
icu4c/source/i18n/unicode/dtrule.h -text
|
||||
icu4c/source/i18n/unicode/rbtz.h -text
|
||||
icu4c/source/i18n/unicode/tzrule.h -text
|
||||
icu4c/source/i18n/unicode/tztrans.h -text
|
||||
icu4c/source/i18n/unicode/vtzone.h -text
|
||||
icu4c/source/i18n/vtzone.cpp -text
|
||||
icu4c/source/samples/layout/cgnomelayout.c -text
|
||||
icu4c/source/samples/layout/gnomeglue.cpp -text
|
||||
icu4c/source/samples/layout/gnomeglue.h -text
|
||||
|
@ -135,6 +147,8 @@ icu4c/source/test/compat/Makefile.in -text
|
|||
icu4c/source/test/compat/readme.txt -text
|
||||
icu4c/source/test/compat/tzdate.c -text
|
||||
icu4c/source/test/compat/tzone.pl -text
|
||||
icu4c/source/test/intltest/tzrulets.cpp -text
|
||||
icu4c/source/test/intltest/tzrulets.h -text
|
||||
icu4c/source/test/perf/charperf/CharPerf_r.pl -text
|
||||
icu4c/source/test/perf/convperf/ConvPerf_r.pl -text
|
||||
icu4c/source/test/perf/normperf/NormPerf_r.pl -text
|
||||
|
|
|
@ -78,7 +78,7 @@ name2uni.o uni2name.o nortrans.o quant.o transreg.o \
|
|||
regexcmp.o rematch.o repattrn.o regexst.o uregex.o uregexc.o \
|
||||
ulocdata.o measfmt.o currfmt.o curramt.o currunit.o measure.o utmscale.o \
|
||||
csdetect.o csmatch.o csr2022.o csrecog.o csrmbcs.o csrsbcs.o csrucode.o csrutf8.o inputext.o \
|
||||
windtfmt.o winnmfmt.o
|
||||
windtfmt.o winnmfmt.o basictz.o dtrule.o rbtz.o tzrule.o tztrans.o vtzone.o
|
||||
|
||||
## Header files to install
|
||||
HEADERS = $(srcdir)/unicode/*.h
|
||||
|
|
519
icu4c/source/i18n/basictz.cpp
Normal file
519
icu4c/source/i18n/basictz.cpp
Normal file
|
@ -0,0 +1,519 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/basictz.h"
|
||||
#include "gregoimp.h"
|
||||
#include "uvector.h"
|
||||
#include "cmemory.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
#define MILLIS_PER_YEAR (365*24*60*60*1000.0)
|
||||
|
||||
BasicTimeZone::BasicTimeZone()
|
||||
: TimeZone() {
|
||||
}
|
||||
|
||||
BasicTimeZone::BasicTimeZone(const UnicodeString &id)
|
||||
: TimeZone(id) {
|
||||
}
|
||||
|
||||
BasicTimeZone::BasicTimeZone(const BasicTimeZone& source)
|
||||
: TimeZone(source) {
|
||||
}
|
||||
|
||||
BasicTimeZone::~BasicTimeZone() {
|
||||
}
|
||||
|
||||
UBool
|
||||
BasicTimeZone::hasEquivalentTransitions(/*const*/ BasicTimeZone& tz, UDate start, UDate end,
|
||||
UBool ignoreDstAmount, UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
if (hasSameRules(tz)) {
|
||||
return TRUE;
|
||||
}
|
||||
// Check the offsets at the start time
|
||||
int32_t raw1, raw2, dst1, dst2;
|
||||
getOffset(start, FALSE, raw1, dst1, status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
tz.getOffset(start, FALSE, raw2, dst2, status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
if (ignoreDstAmount) {
|
||||
if ((raw1 + dst1 != raw2 + dst2)
|
||||
|| (dst1 != 0 && dst2 == 0)
|
||||
|| (dst1 == 0 && dst2 != 0)) {
|
||||
return FALSE;
|
||||
}
|
||||
} else {
|
||||
if (raw1 != raw2 || dst1 != dst2) {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
// Check transitions in the range
|
||||
UDate time = start;
|
||||
TimeZoneTransition tr1, tr2;
|
||||
while (TRUE) {
|
||||
UBool avail1 = getNextTransition(time, FALSE, tr1);
|
||||
UBool avail2 = tz.getNextTransition(time, FALSE, tr2);
|
||||
|
||||
if (ignoreDstAmount) {
|
||||
// Skip a transition which only differ the amount of DST savings
|
||||
if (avail1
|
||||
&& (tr1.getFrom()->getRawOffset() + tr1.getFrom()->getDSTSavings()
|
||||
== tr1.getTo()->getRawOffset() + tr1.getTo()->getDSTSavings())
|
||||
&& (tr1.getFrom()->getDSTSavings() != 0 && tr1.getTo()->getDSTSavings() != 0)) {
|
||||
getNextTransition(tr1.getTime(), FALSE, tr1);
|
||||
}
|
||||
if (avail2
|
||||
&& (tr2.getFrom()->getRawOffset() + tr2.getFrom()->getDSTSavings()
|
||||
== tr2.getTo()->getRawOffset() + tr2.getTo()->getDSTSavings())
|
||||
&& (tr2.getFrom()->getDSTSavings() != 0 && tr2.getTo()->getDSTSavings() != 0)) {
|
||||
getNextTransition(tr2.getTime(), FALSE, tr2);
|
||||
}
|
||||
}
|
||||
|
||||
UBool inRange1 = (avail1 && tr1.getTime() <= end);
|
||||
UBool inRange2 = (avail2 && tr2.getTime() <= end);
|
||||
if (!inRange1 && !inRange2) {
|
||||
// No more transition in the range
|
||||
break;
|
||||
}
|
||||
if (!inRange1 || !inRange2) {
|
||||
return FALSE;
|
||||
}
|
||||
if (tr1.getTime() != tr2.getTime()) {
|
||||
return FALSE;
|
||||
}
|
||||
if (ignoreDstAmount) {
|
||||
if (tr1.getTo()->getRawOffset() + tr1.getTo()->getDSTSavings()
|
||||
!= tr2.getTo()->getRawOffset() + tr2.getTo()->getDSTSavings()
|
||||
|| tr1.getTo()->getDSTSavings() != 0 && tr2.getTo()->getDSTSavings() == 0
|
||||
|| tr1.getTo()->getDSTSavings() == 0 && tr2.getTo()->getDSTSavings() != 0) {
|
||||
return FALSE;
|
||||
}
|
||||
} else {
|
||||
if (tr1.getTo()->getRawOffset() != tr2.getTo()->getRawOffset() ||
|
||||
tr1.getTo()->getDSTSavings() != tr2.getTo()->getDSTSavings()) {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
time = tr1.getTime();
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
void
|
||||
BasicTimeZone::getSimpleRulesNear(UDate date, InitialTimeZoneRule*& initial,
|
||||
AnnualTimeZoneRule*& std, AnnualTimeZoneRule*& dst, UErrorCode& status) /*const*/ {
|
||||
initial = NULL;
|
||||
std = NULL;
|
||||
dst = NULL;
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
int32_t initialRaw, initialDst;
|
||||
UnicodeString initialName;
|
||||
|
||||
AnnualTimeZoneRule *ar1 = NULL;
|
||||
AnnualTimeZoneRule *ar2 = NULL;
|
||||
UnicodeString name;
|
||||
|
||||
UBool avail;
|
||||
TimeZoneTransition tr;
|
||||
// Get the next transition
|
||||
avail = getNextTransition(date, FALSE, tr);
|
||||
if (avail) {
|
||||
tr.getFrom()->getName(initialName);
|
||||
initialRaw = tr.getFrom()->getRawOffset();
|
||||
initialDst = tr.getFrom()->getDSTSavings();
|
||||
|
||||
// Check if the next transition is either DST->STD or STD->DST and
|
||||
// within roughly 1 year from the specified date
|
||||
UDate nextTransitionTime = tr.getTime();
|
||||
if (((tr.getFrom()->getDSTSavings() == 0 && tr.getTo()->getDSTSavings() != 0)
|
||||
|| (tr.getFrom()->getDSTSavings() != 0 && tr.getTo()->getDSTSavings() == 0))
|
||||
&& (date + MILLIS_PER_YEAR > nextTransitionTime)) {
|
||||
|
||||
int32_t year, month, dom, dow, doy, mid;
|
||||
UDate d;
|
||||
|
||||
// Get local wall time for the next transition time
|
||||
Grego::timeToFields(nextTransitionTime + initialRaw + initialDst,
|
||||
year, month, dom, dow, doy, mid);
|
||||
int32_t weekInMonth = Grego::dayOfWeekInMonth(year, month, dom);
|
||||
// Create DOW rule
|
||||
DateTimeRule *dtr = new DateTimeRule(month, weekInMonth, dow, mid, DateTimeRule::WALL_TIME);
|
||||
tr.getTo()->getName(name);
|
||||
ar1 = new AnnualTimeZoneRule(name, tr.getTo()->getRawOffset(), tr.getTo()->getDSTSavings(),
|
||||
dtr, year, AnnualTimeZoneRule::MAX_YEAR);
|
||||
|
||||
// Get the next next transition
|
||||
avail = getNextTransition(nextTransitionTime, FALSE, tr);
|
||||
if (avail) {
|
||||
// Check if the next next transition is either DST->STD or STD->DST
|
||||
// and within roughly 1 year from the next transition
|
||||
if (((tr.getFrom()->getDSTSavings() == 0 && tr.getTo()->getDSTSavings() != 0)
|
||||
|| (tr.getFrom()->getDSTSavings() != 0 && tr.getTo()->getDSTSavings() == 0))
|
||||
&& nextTransitionTime + MILLIS_PER_YEAR > tr.getTime()) {
|
||||
|
||||
// Get local wall time for the next transition time
|
||||
Grego::timeToFields(tr.getTime() + tr.getFrom()->getRawOffset() + tr.getFrom()->getDSTSavings(),
|
||||
year, month, dom, dow, doy, mid);
|
||||
weekInMonth = Grego::dayOfWeekInMonth(year, month, dom);
|
||||
// Generate another DOW rule
|
||||
dtr = new DateTimeRule(month, weekInMonth, dow, mid, DateTimeRule::WALL_TIME);
|
||||
tr.getTo()->getName(name);
|
||||
ar2 = new AnnualTimeZoneRule(name, tr.getTo()->getRawOffset(), tr.getTo()->getDSTSavings(),
|
||||
dtr, year - 1, AnnualTimeZoneRule::MAX_YEAR);
|
||||
|
||||
// Make sure this rule can be applied to the specified date
|
||||
avail = ar2->getPreviousStart(date, tr.getFrom()->getRawOffset(), tr.getFrom()->getDSTSavings(), TRUE, d);
|
||||
if (!avail || d > date
|
||||
|| initialRaw != tr.getTo()->getRawOffset()
|
||||
|| initialDst != tr.getTo()->getDSTSavings()) {
|
||||
// We cannot use this rule as the second transition rule
|
||||
delete ar2;
|
||||
ar2 = NULL;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ar2 == NULL) {
|
||||
// Try previous transition
|
||||
avail = getPreviousTransition(date, TRUE, tr);
|
||||
if (avail) {
|
||||
// Check if the previous transition is either DST->STD or STD->DST.
|
||||
// The actual transition time does not matter here.
|
||||
if ((tr.getFrom()->getDSTSavings() == 0 && tr.getTo()->getDSTSavings() != 0)
|
||||
|| (tr.getFrom()->getDSTSavings() != 0 && tr.getTo()->getDSTSavings() == 0)) {
|
||||
|
||||
// Generate another DOW rule
|
||||
Grego::timeToFields(tr.getTime() + tr.getFrom()->getRawOffset() + tr.getFrom()->getDSTSavings(),
|
||||
year, month, dom, dow, doy, mid);
|
||||
weekInMonth = Grego::dayOfWeekInMonth(year, month, dom);
|
||||
dtr = new DateTimeRule(month, weekInMonth, dow, mid, DateTimeRule::WALL_TIME);
|
||||
tr.getTo()->getName(name);
|
||||
ar2 = new AnnualTimeZoneRule(name, tr.getTo()->getRawOffset(), tr.getTo()->getDSTSavings(),
|
||||
dtr, ar1->getStartYear() - 1, AnnualTimeZoneRule::MAX_YEAR);
|
||||
|
||||
// Check if this rule start after the first rule after the specified date
|
||||
avail = ar2->getNextStart(date, tr.getFrom()->getRawOffset(), tr.getFrom()->getDSTSavings(), FALSE, d);
|
||||
if (!avail || d <= nextTransitionTime) {
|
||||
// We cannot use this rule as the second transition rule
|
||||
delete ar2;
|
||||
ar2 = NULL;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ar2 == NULL) {
|
||||
// Cannot find a good pair of AnnualTimeZoneRule
|
||||
delete ar1;
|
||||
ar1 = NULL;
|
||||
} else {
|
||||
// The initial rule should represent the rule before the previous transition
|
||||
ar1->getName(initialName);
|
||||
initialRaw = ar1->getRawOffset();
|
||||
initialDst = ar1->getDSTSavings();
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// Try the previous one
|
||||
avail = getPreviousTransition(date, TRUE, tr);
|
||||
if (avail) {
|
||||
tr.getTo()->getName(initialName);
|
||||
initialRaw = tr.getTo()->getRawOffset();
|
||||
initialDst = tr.getTo()->getDSTSavings();
|
||||
} else {
|
||||
// No transitions in the past. Just use the current offsets
|
||||
getOffset(date, FALSE, initialRaw, initialDst, status);
|
||||
if (U_FAILURE(status)) {
|
||||
if (ar1 != NULL) {
|
||||
delete ar1;
|
||||
}
|
||||
if (ar2 != NULL) {
|
||||
delete ar2;
|
||||
}
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
// Set the initial rule
|
||||
initial = new InitialTimeZoneRule(initialName, initialRaw, initialDst);
|
||||
|
||||
// Set the standard and daylight saving rules
|
||||
if (ar1 != NULL && ar2 != NULL) {
|
||||
if (ar1->getDSTSavings() != 0) {
|
||||
dst = ar1;
|
||||
std = ar2;
|
||||
} else {
|
||||
std = ar1;
|
||||
dst = ar2;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
BasicTimeZone::getTimeZoneRulesAfter(UDate start, InitialTimeZoneRule*& initial,
|
||||
UVector*& transitionRules, UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
|
||||
const InitialTimeZoneRule *orgini;
|
||||
const TimeZoneRule **orgtrs = NULL;
|
||||
TimeZoneTransition tzt;
|
||||
UBool avail;
|
||||
UVector *orgRules = NULL;
|
||||
int32_t ruleCount;
|
||||
TimeZoneRule *r = NULL;
|
||||
UBool *done = NULL;
|
||||
InitialTimeZoneRule *res_initial = NULL;
|
||||
UVector *filteredRules = NULL;
|
||||
UnicodeString name;
|
||||
int32_t i;
|
||||
UDate time, t;
|
||||
UDate *newTimes = NULL;
|
||||
UDate firstStart;
|
||||
UBool bFinalStd = FALSE, bFinalDst = FALSE;
|
||||
|
||||
// Original transition rules
|
||||
ruleCount = countTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
orgRules = new UVector(ruleCount, status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
orgtrs = (const TimeZoneRule**)uprv_malloc(sizeof(TimeZoneRule*)*ruleCount);
|
||||
if (orgtrs == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto error;
|
||||
}
|
||||
getTimeZoneRules(orgini, orgtrs, ruleCount, status);
|
||||
if (U_FAILURE(status)) {
|
||||
delete orgtrs;
|
||||
goto error;
|
||||
}
|
||||
for (i = 0; i < ruleCount; i++) {
|
||||
orgRules->addElement(orgtrs[i]->clone(), status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
uprv_free(orgtrs);
|
||||
|
||||
avail = getPreviousTransition(start, TRUE, tzt);
|
||||
if (!avail) {
|
||||
// No need to filter out rules only applicable to time before the start
|
||||
initial = orgini->clone();
|
||||
transitionRules = orgRules;
|
||||
return;
|
||||
}
|
||||
|
||||
done = (UBool*)uprv_malloc(sizeof(UBool)*ruleCount);
|
||||
if (done == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto error;
|
||||
}
|
||||
filteredRules = new UVector(status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
|
||||
// Create initial rule
|
||||
tzt.getTo()->getName(name);
|
||||
res_initial = new InitialTimeZoneRule(name, tzt.getTo()->getRawOffset(),
|
||||
tzt.getTo()->getDSTSavings());
|
||||
|
||||
// Mark rules which does not need to be processed
|
||||
for (i = 0; i < ruleCount; i++) {
|
||||
r = (TimeZoneRule*)orgRules->elementAt(i);
|
||||
avail = r->getNextStart(start, res_initial->getRawOffset(), res_initial->getDSTSavings(), FALSE, time);
|
||||
done[i] = !avail;
|
||||
}
|
||||
|
||||
time = start;
|
||||
while (!bFinalStd || !bFinalDst) {
|
||||
avail = getNextTransition(time, FALSE, tzt);
|
||||
if (!avail) {
|
||||
break;
|
||||
}
|
||||
time = tzt.getTime();
|
||||
|
||||
const TimeZoneRule *toRule = tzt.getTo();
|
||||
for (i = 0; i < ruleCount; i++) {
|
||||
r = (TimeZoneRule*)orgRules->elementAt(i);
|
||||
if (*r == *toRule) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (i >= ruleCount) {
|
||||
// This case should never happen
|
||||
status = U_INVALID_STATE_ERROR;
|
||||
goto error;
|
||||
}
|
||||
if (done[i]) {
|
||||
continue;
|
||||
}
|
||||
if (toRule->getDynamicClassID() == TimeArrayTimeZoneRule::getStaticClassID()) {
|
||||
TimeArrayTimeZoneRule *tar = (TimeArrayTimeZoneRule*)toRule;
|
||||
|
||||
// Get the previous raw offset and DST savings before the very first start time
|
||||
TimeZoneTransition tzt0;
|
||||
t = start;
|
||||
while (TRUE) {
|
||||
avail = getNextTransition(t, FALSE, tzt0);
|
||||
if (!avail) {
|
||||
break;
|
||||
}
|
||||
if (*(tzt0.getTo()) == *tar) {
|
||||
break;
|
||||
}
|
||||
t = tzt0.getTime();
|
||||
}
|
||||
if (avail) {
|
||||
// Check if the entire start times to be added
|
||||
tar->getFirstStart(tzt.getFrom()->getRawOffset(), tzt.getFrom()->getDSTSavings(), firstStart);
|
||||
if (firstStart > start) {
|
||||
// Just add the rule as is
|
||||
filteredRules->addElement(tar->clone(), status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
} else {
|
||||
// Colllect transitions after the start time
|
||||
int32_t startTimes;
|
||||
DateTimeRule::TimeRuleType timeType;
|
||||
int32_t idx;
|
||||
|
||||
startTimes = tar->countStartTimes();
|
||||
timeType = tar->getTimeType();
|
||||
for (idx = 0; idx < startTimes; idx++) {
|
||||
tar->getStartTimeAt(idx, t);
|
||||
if (timeType == DateTimeRule::STANDARD_TIME) {
|
||||
t -= tzt.getFrom()->getRawOffset();
|
||||
}
|
||||
if (timeType == DateTimeRule::WALL_TIME) {
|
||||
t -= tzt.getFrom()->getDSTSavings();
|
||||
}
|
||||
if (t > start) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
int32_t asize = startTimes - idx;
|
||||
if (asize > 0) {
|
||||
newTimes = (UDate*)uprv_malloc(sizeof(UDate) * asize);
|
||||
if (newTimes == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto error;
|
||||
}
|
||||
for (int32_t newidx = 0; newidx < asize; newidx++) {
|
||||
tar->getStartTimeAt(idx + newidx, newTimes[newidx]);
|
||||
if (U_FAILURE(status)) {
|
||||
uprv_free(newTimes);
|
||||
newTimes = NULL;
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
tar->getName(name);
|
||||
TimeArrayTimeZoneRule *newTar = new TimeArrayTimeZoneRule(name,
|
||||
tar->getRawOffset(), tar->getDSTSavings(), newTimes, asize, timeType);
|
||||
uprv_free(newTimes);
|
||||
filteredRules->addElement(newTar, status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (toRule->getDynamicClassID() == AnnualTimeZoneRule::getStaticClassID()) {
|
||||
AnnualTimeZoneRule *ar = (AnnualTimeZoneRule*)toRule;
|
||||
ar->getFirstStart(tzt.getFrom()->getRawOffset(), tzt.getFrom()->getDSTSavings(), firstStart);
|
||||
if (firstStart == tzt.getTime()) {
|
||||
// Just add the rule as is
|
||||
filteredRules->addElement(ar->clone(), status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
} else {
|
||||
// Calculate the transition year
|
||||
int32_t year, month, dom, dow, doy, mid;
|
||||
Grego::timeToFields(tzt.getTime(), year, month, dom, dow, doy, mid);
|
||||
// Re-create the rule
|
||||
ar->getName(name);
|
||||
AnnualTimeZoneRule *newAr = new AnnualTimeZoneRule(name, ar->getRawOffset(), ar->getDSTSavings(),
|
||||
*(ar->getRule()), year, ar->getEndYear());
|
||||
filteredRules->addElement(newAr, status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
// check if this is a final rule
|
||||
if (ar->getEndYear() == AnnualTimeZoneRule::MAX_YEAR) {
|
||||
// After bot final standard and dst rules are processed,
|
||||
// exit this while loop.
|
||||
if (ar->getDSTSavings() == 0) {
|
||||
bFinalStd = TRUE;
|
||||
} else {
|
||||
bFinalDst = TRUE;
|
||||
}
|
||||
}
|
||||
}
|
||||
done[i] = TRUE;
|
||||
}
|
||||
|
||||
// Set the results
|
||||
if (orgRules != NULL) {
|
||||
while (!orgRules->isEmpty()) {
|
||||
r = (TimeZoneRule*)orgRules->orphanElementAt(0);
|
||||
delete r;
|
||||
}
|
||||
delete orgRules;
|
||||
}
|
||||
if (done != NULL) {
|
||||
uprv_free(done);
|
||||
}
|
||||
|
||||
initial = res_initial;
|
||||
transitionRules = filteredRules;
|
||||
return;
|
||||
|
||||
error:
|
||||
if (orgtrs != NULL) {
|
||||
uprv_free(orgtrs);
|
||||
}
|
||||
if (orgRules != NULL) {
|
||||
while (!orgRules->isEmpty()) {
|
||||
r = (TimeZoneRule*)orgRules->orphanElementAt(0);
|
||||
delete r;
|
||||
}
|
||||
delete orgRules;
|
||||
}
|
||||
if (done != NULL) {
|
||||
uprv_free(done);
|
||||
}
|
||||
|
||||
initial = NULL;
|
||||
transitionRules = NULL;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
//eof
|
139
icu4c/source/i18n/dtrule.cpp
Normal file
139
icu4c/source/i18n/dtrule.cpp
Normal file
|
@ -0,0 +1,139 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/dtrule.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(DateTimeRule)
|
||||
|
||||
DateTimeRule::DateTimeRule(int32_t month,
|
||||
int32_t dayOfMonth,
|
||||
int32_t millisInDay,
|
||||
TimeRuleType timeType)
|
||||
: UObject(),
|
||||
fMonth(month), fDayOfMonth(dayOfMonth), fDayOfWeek(0), fWeekInMonth(0), fMillisInDay(millisInDay),
|
||||
fDateRuleType(DateTimeRule::DOM), fTimeRuleType(timeType) {
|
||||
}
|
||||
|
||||
DateTimeRule::DateTimeRule(int32_t month,
|
||||
int32_t weekInMonth,
|
||||
int32_t dayOfWeek,
|
||||
int32_t millisInDay,
|
||||
TimeRuleType timeType)
|
||||
: UObject(),
|
||||
fMonth(month), fDayOfMonth(0), fDayOfWeek(dayOfWeek), fWeekInMonth(weekInMonth), fMillisInDay(millisInDay),
|
||||
fDateRuleType(DateTimeRule::DOW), fTimeRuleType(timeType) {
|
||||
}
|
||||
|
||||
DateTimeRule::DateTimeRule(int32_t month,
|
||||
int32_t dayOfMonth,
|
||||
int32_t dayOfWeek,
|
||||
UBool after,
|
||||
int32_t millisInDay,
|
||||
TimeRuleType timeType)
|
||||
: UObject(),
|
||||
fMonth(month), fDayOfMonth(dayOfMonth), fDayOfWeek(dayOfWeek), fWeekInMonth(0), fMillisInDay(millisInDay),
|
||||
fTimeRuleType(timeType) {
|
||||
if (after) {
|
||||
fDateRuleType = DateTimeRule::DOW_GEQ_DOM;
|
||||
} else {
|
||||
fDateRuleType = DateTimeRule::DOW_LEQ_DOM;
|
||||
}
|
||||
}
|
||||
|
||||
DateTimeRule::DateTimeRule(const DateTimeRule& source)
|
||||
: UObject(source),
|
||||
fMonth(source.fMonth), fDayOfMonth(source.fDayOfMonth), fDayOfWeek(source.fDayOfWeek),
|
||||
fWeekInMonth(source.fWeekInMonth), fMillisInDay(source.fMillisInDay),
|
||||
fDateRuleType(source.fDateRuleType), fTimeRuleType(source.fTimeRuleType) {
|
||||
}
|
||||
|
||||
DateTimeRule::~DateTimeRule() {
|
||||
}
|
||||
|
||||
DateTimeRule*
|
||||
DateTimeRule::clone() const {
|
||||
return new DateTimeRule(*this);
|
||||
}
|
||||
|
||||
DateTimeRule&
|
||||
DateTimeRule::operator=(const DateTimeRule& right) {
|
||||
if (this != &right) {
|
||||
fMonth = right.fMonth;
|
||||
fDayOfMonth = right.fDayOfMonth;
|
||||
fDayOfWeek = right.fDayOfWeek;
|
||||
fWeekInMonth = right.fWeekInMonth;
|
||||
fMillisInDay = right.fMillisInDay;
|
||||
fDateRuleType = right.fDateRuleType;
|
||||
fTimeRuleType = right.fTimeRuleType;
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
DateTimeRule::operator==(const DateTimeRule& that) const {
|
||||
return ((this == &that) ||
|
||||
(getDynamicClassID() == that.getDynamicClassID() &&
|
||||
fMonth == that.fMonth &&
|
||||
fDayOfMonth == that.fDayOfMonth &&
|
||||
fDayOfWeek == that.fDayOfWeek &&
|
||||
fWeekInMonth == that.fWeekInMonth &&
|
||||
fMillisInDay == that.fMillisInDay &&
|
||||
fDateRuleType == that.fDateRuleType &&
|
||||
fTimeRuleType == that.fTimeRuleType));
|
||||
}
|
||||
|
||||
UBool
|
||||
DateTimeRule::operator!=(const DateTimeRule& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
DateTimeRule::DateRuleType
|
||||
DateTimeRule::getDateRuleType(void) const {
|
||||
return fDateRuleType;
|
||||
}
|
||||
|
||||
DateTimeRule::TimeRuleType
|
||||
DateTimeRule::getTimeRuleType(void) const {
|
||||
return fTimeRuleType;
|
||||
}
|
||||
|
||||
int32_t
|
||||
DateTimeRule::getRuleMonth(void) const {
|
||||
return fMonth;
|
||||
}
|
||||
|
||||
int32_t
|
||||
DateTimeRule::getRuleDayOfMonth(void) const {
|
||||
return fDayOfMonth;
|
||||
}
|
||||
|
||||
int32_t
|
||||
DateTimeRule::getRuleDayOfWeek(void) const {
|
||||
return fDayOfWeek;
|
||||
}
|
||||
|
||||
int32_t
|
||||
DateTimeRule::getRuleWeekInMonth(void) const {
|
||||
return fWeekInMonth;
|
||||
}
|
||||
|
||||
int32_t
|
||||
DateTimeRule::getRuleMillisInDay(void) const {
|
||||
return fMillisInDay;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
//eof
|
|
@ -130,6 +130,32 @@ void Grego::dayToFields(double day, int32_t& year, int32_t& month,
|
|||
doy++; // one-based doy
|
||||
}
|
||||
|
||||
void Grego::timeToFields(UDate time, int32_t& year, int32_t& month,
|
||||
int32_t& dom, int32_t& dow, int32_t& doy, int32_t& mid) {
|
||||
double millisInDay;
|
||||
double day = Math::floorDivide((double)time, (double)U_MILLIS_PER_DAY, millisInDay);
|
||||
mid = (int32_t)millisInDay;
|
||||
dayToFields(day, year, month, dom, dow, doy);
|
||||
}
|
||||
|
||||
int32_t Grego::dayOfWeek(double day) {
|
||||
int32_t dow;
|
||||
Math::floorDivide(day + UCAL_THURSDAY, 7, dow);
|
||||
return (dow == 0) ? UCAL_SATURDAY : dow;
|
||||
}
|
||||
|
||||
int32_t Grego::dayOfWeekInMonth(int32_t year, int32_t month, int32_t dom) {
|
||||
int32_t weekInMonth = (dom + 6)/7;
|
||||
if (weekInMonth == 4) {
|
||||
if (dom + 7 > monthLength(year, month)) {
|
||||
weekInMonth = -1;
|
||||
}
|
||||
} else if (weekInMonth == 5) {
|
||||
weekInMonth = -1;
|
||||
}
|
||||
return weekInMonth;
|
||||
}
|
||||
|
||||
/* ---- CalendarData ------ */
|
||||
|
||||
#define U_CALENDAR_KEY "calendar"
|
||||
|
|
|
@ -190,6 +190,37 @@ class Grego {
|
|||
static inline void dayToFields(double day, int32_t& year, int32_t& month,
|
||||
int32_t& dom, int32_t& dow);
|
||||
|
||||
/**
|
||||
* Convert a 1970-epoch milliseconds to proleptic Gregorian year,
|
||||
* month, day-of-month, and day-of-week, day of year and millis-in-day.
|
||||
* @param time 1970-epoch milliseconds
|
||||
* @param year output parameter to receive year
|
||||
* @param month output parameter to receive month (0-based, 0==Jan)
|
||||
* @param dom output parameter to receive day-of-month (1-based)
|
||||
* @param dow output parameter to receive day-of-week (1-based, 1==Sun)
|
||||
* @param doy output parameter to receive day-of-year (1-based)
|
||||
* @param mid output parameter to recieve millis-in-day
|
||||
*/
|
||||
static void timeToFields(UDate time, int32_t& year, int32_t& month,
|
||||
int32_t& dom, int32_t& dow, int32_t& doy, int32_t& mid);
|
||||
|
||||
/**
|
||||
* Return the day of week on the 1970-epoch day
|
||||
* @param day the 1970-epoch day (integral value)
|
||||
* @return the day of week
|
||||
*/
|
||||
static int32_t dayOfWeek(double day);
|
||||
|
||||
/**
|
||||
* Returns the ordinal number for the specified day of week within the month.
|
||||
* The valid return value is 1, 2, 3, 4 or -1.
|
||||
* @param year Gregorian year, with 0 == 1 BCE, -1 == 2 BCE, etc.
|
||||
* @param month 0-based month, with 0==Jan
|
||||
* @param dom 1-based day of month
|
||||
* @return The ordinal number for the specified day of week within the month
|
||||
*/
|
||||
static int32_t dayOfWeekInMonth(int32_t year, int32_t month, int32_t dom);
|
||||
|
||||
/**
|
||||
* Converts Julian day to time as milliseconds.
|
||||
* @param julian the given Julian day number.
|
||||
|
|
|
@ -522,6 +522,14 @@
|
|||
RelativePath=".\astro.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\basictz.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\basictz.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\buddhcal.cpp"
|
||||
>
|
||||
|
@ -754,6 +762,14 @@
|
|||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\dtrule.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\dtrule.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\fieldpos.h"
|
||||
>
|
||||
|
@ -1086,6 +1102,14 @@
|
|||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\rbtz.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\rbtz.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\reldtfmt.cpp"
|
||||
>
|
||||
|
@ -1172,6 +1196,22 @@
|
|||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\tzrule.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\tzrule.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\tztrans.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\tztrans.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\ucal.cpp"
|
||||
>
|
||||
|
@ -1353,6 +1393,14 @@
|
|||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\vtzone.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\unicode\vtzone.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\windtfmt.cpp"
|
||||
>
|
||||
|
|
|
@ -19,6 +19,7 @@
|
|||
#include "gregoimp.h"
|
||||
#include "cmemory.h"
|
||||
#include "uassert.h"
|
||||
#include "uvector.h"
|
||||
#include <float.h> // DBL_MAX
|
||||
|
||||
#ifdef U_DEBUG_TZ
|
||||
|
@ -55,7 +56,8 @@ UOBJECT_DEFINE_RTTI_IMPLEMENTATION(OlsonTimeZone)
|
|||
* Default constructor. Creates a time zone with an empty ID and
|
||||
* a fixed GMT offset of zero.
|
||||
*/
|
||||
/*OlsonTimeZone::OlsonTimeZone() : finalYear(INT32_MAX), finalMillis(DBL_MAX), finalZone(0) {
|
||||
/*OlsonTimeZone::OlsonTimeZone() : finalYear(INT32_MAX), finalMillis(DBL_MAX), finalZone(0), transitionRulesInitialized(FALSE) {
|
||||
clearTransitionRules();
|
||||
constructEmpty();
|
||||
}*/
|
||||
|
||||
|
@ -80,8 +82,9 @@ void OlsonTimeZone::constructEmpty() {
|
|||
OlsonTimeZone::OlsonTimeZone(const UResourceBundle* top,
|
||||
const UResourceBundle* res,
|
||||
UErrorCode& ec) :
|
||||
finalYear(INT32_MAX), finalMillis(DBL_MAX), finalZone(0)
|
||||
finalYear(INT32_MAX), finalMillis(DBL_MAX), finalZone(0), transitionRulesInitialized(FALSE)
|
||||
{
|
||||
clearTransitionRules();
|
||||
U_DEBUG_TZ_MSG(("OlsonTimeZone(%s)\n", ures_getKey((UResourceBundle*)res)));
|
||||
if ((top == NULL || res == NULL) && U_SUCCESS(ec)) {
|
||||
ec = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
|
@ -213,7 +216,7 @@ OlsonTimeZone::OlsonTimeZone(const UResourceBundle* top,
|
|||
* Copy constructor
|
||||
*/
|
||||
OlsonTimeZone::OlsonTimeZone(const OlsonTimeZone& other) :
|
||||
TimeZone(other), finalZone(0) {
|
||||
BasicTimeZone(other), finalZone(0) {
|
||||
*this = other;
|
||||
}
|
||||
|
||||
|
@ -231,6 +234,7 @@ OlsonTimeZone& OlsonTimeZone::operator=(const OlsonTimeZone& other) {
|
|||
delete finalZone;
|
||||
finalZone = (other.finalZone != 0) ?
|
||||
(SimpleTimeZone*) other.finalZone->clone() : 0;
|
||||
clearTransitionRules();
|
||||
return *this;
|
||||
}
|
||||
|
||||
|
@ -238,6 +242,7 @@ OlsonTimeZone& OlsonTimeZone::operator=(const OlsonTimeZone& other) {
|
|||
* Destructor
|
||||
*/
|
||||
OlsonTimeZone::~OlsonTimeZone() {
|
||||
deleteTransitionRules();
|
||||
delete finalZone;
|
||||
}
|
||||
|
||||
|
@ -245,29 +250,10 @@ OlsonTimeZone::~OlsonTimeZone() {
|
|||
* Returns true if the two TimeZone objects are equal.
|
||||
*/
|
||||
UBool OlsonTimeZone::operator==(const TimeZone& other) const {
|
||||
const OlsonTimeZone* z = (const OlsonTimeZone*) &other;
|
||||
|
||||
return TimeZone::operator==(other) &&
|
||||
// [sic] pointer comparison: typeData points into
|
||||
// memory-mapped or DLL space, so if two zones have the same
|
||||
// pointer, they are equal.
|
||||
(typeData == z->typeData ||
|
||||
// If the pointers are not equal, the zones may still
|
||||
// be equal if their rules and transitions are equal
|
||||
(finalYear == z->finalYear &&
|
||||
// Don't compare finalMillis; if finalYear is ==, so is finalMillis
|
||||
((finalZone == 0 && z->finalZone == 0) ||
|
||||
(finalZone != 0 && z->finalZone != 0 &&
|
||||
*finalZone == *z->finalZone)) &&
|
||||
transitionCount == z->transitionCount &&
|
||||
typeCount == z->typeCount &&
|
||||
uprv_memcmp(transitionTimes, z->transitionTimes,
|
||||
sizeof(transitionTimes[0]) * transitionCount) == 0 &&
|
||||
uprv_memcmp(typeOffsets, z->typeOffsets,
|
||||
(sizeof(typeOffsets[0]) * typeCount) << 1) == 0 &&
|
||||
uprv_memcmp(typeData, z->typeData,
|
||||
(sizeof(typeData[0]) * typeCount)) == 0
|
||||
));
|
||||
return ((this == &other) ||
|
||||
(getDynamicClassID() == other.getDynamicClassID() &&
|
||||
TimeZone::operator==(other) &&
|
||||
hasSameRules(other)));
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -546,6 +532,413 @@ UBool OlsonTimeZone::inDaylightTime(UDate date, UErrorCode& ec) const {
|
|||
return dst != 0;
|
||||
}
|
||||
|
||||
UBool
|
||||
OlsonTimeZone::hasSameRules(const TimeZone &other) const {
|
||||
if (this == &other) {
|
||||
return TRUE;
|
||||
}
|
||||
if (other.getDynamicClassID() != OlsonTimeZone::getStaticClassID()) {
|
||||
return FALSE;
|
||||
}
|
||||
const OlsonTimeZone* z = (const OlsonTimeZone*) &other;
|
||||
|
||||
// [sic] pointer comparison: typeData points into
|
||||
// memory-mapped or DLL space, so if two zones have the same
|
||||
// pointer, they are equal.
|
||||
if (typeData == z->typeData) {
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
// If the pointers are not equal, the zones may still
|
||||
// be equal if their rules and transitions are equal
|
||||
return
|
||||
(finalYear == z->finalYear &&
|
||||
// Don't compare finalMillis; if finalYear is ==, so is finalMillis
|
||||
((finalZone == 0 && z->finalZone == 0) ||
|
||||
(finalZone != 0 && z->finalZone != 0 && *finalZone == *z->finalZone)) &&
|
||||
|
||||
transitionCount == z->transitionCount &&
|
||||
typeCount == z->typeCount &&
|
||||
uprv_memcmp(transitionTimes, z->transitionTimes,
|
||||
sizeof(transitionTimes[0]) * transitionCount) == 0 &&
|
||||
uprv_memcmp(typeOffsets, z->typeOffsets,
|
||||
(sizeof(typeOffsets[0]) * typeCount) << 1) == 0 &&
|
||||
uprv_memcmp(typeData, z->typeData,
|
||||
(sizeof(typeData[0]) * typeCount)) == 0);
|
||||
}
|
||||
|
||||
void
|
||||
OlsonTimeZone::clearTransitionRules(void) {
|
||||
initialRule = NULL;
|
||||
firstTZTransition = NULL;
|
||||
firstFinalTZTransition = NULL;
|
||||
historicRules = NULL;
|
||||
historicRuleCount = 0;
|
||||
finalZoneWithStartYear = NULL;
|
||||
firstTZTransitionIdx = 0;
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
void
|
||||
OlsonTimeZone::deleteTransitionRules(void) {
|
||||
if (initialRule != NULL) {
|
||||
delete initialRule;
|
||||
}
|
||||
if (firstTZTransition != NULL) {
|
||||
delete firstTZTransition;
|
||||
}
|
||||
if (firstFinalTZTransition != NULL) {
|
||||
delete firstFinalTZTransition;
|
||||
}
|
||||
if (finalZoneWithStartYear != NULL) {
|
||||
delete finalZoneWithStartYear;
|
||||
}
|
||||
if (historicRules != NULL) {
|
||||
for (int i = 0; i < historicRuleCount; i++) {
|
||||
if (historicRules[i] != NULL) {
|
||||
delete historicRules[i];
|
||||
}
|
||||
}
|
||||
uprv_free(historicRules);
|
||||
}
|
||||
clearTransitionRules();
|
||||
}
|
||||
|
||||
void
|
||||
OlsonTimeZone::initTransitionRules(UErrorCode& status) {
|
||||
if(U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
if (transitionRulesInitialized) {
|
||||
return;
|
||||
}
|
||||
deleteTransitionRules();
|
||||
UnicodeString tzid;
|
||||
getID(tzid);
|
||||
|
||||
UnicodeString stdName = tzid + "(STD)";
|
||||
UnicodeString dstName = tzid + "(DST)";
|
||||
|
||||
int32_t raw, dst;
|
||||
if (transitionCount > 0) {
|
||||
int16_t transitionIdx, typeIdx;
|
||||
|
||||
// Note: Since 2007c, the very first transition data is a dummy entry
|
||||
// added for resolving a offset calculation problem.
|
||||
|
||||
// Create initial rule
|
||||
typeIdx = (int16_t)typeData[0]; // initial type
|
||||
raw = rawOffset(typeIdx) * U_MILLIS_PER_SECOND;
|
||||
dst = dstOffset(typeIdx) * U_MILLIS_PER_SECOND;
|
||||
initialRule = new InitialTimeZoneRule((dst == 0 ? stdName : dstName), raw, dst);
|
||||
|
||||
firstTZTransitionIdx = 0;
|
||||
for (transitionIdx = 1; transitionIdx < transitionCount; transitionIdx++) {
|
||||
firstTZTransitionIdx++;
|
||||
if (typeIdx != (int16_t)typeData[transitionIdx]) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (transitionIdx == transitionCount) {
|
||||
// Actually no transitions...
|
||||
} else {
|
||||
// Build historic rule array
|
||||
UDate* times = (UDate*)uprv_malloc(sizeof(UDate)*transitionCount); /* large enough to store all transition times */
|
||||
if (times == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
deleteTransitionRules();
|
||||
return;
|
||||
}
|
||||
for (typeIdx = 0; typeIdx < typeCount; typeIdx++) {
|
||||
// Gather all start times for each pair of offsets
|
||||
int32_t nTimes = 0;
|
||||
for (transitionIdx = firstTZTransitionIdx; transitionIdx < transitionCount; transitionIdx++) {
|
||||
if (typeIdx == (int16_t)typeData[transitionIdx]) {
|
||||
UDate tt = ((UDate)transitionTimes[transitionIdx]) * U_MILLIS_PER_SECOND;
|
||||
if (tt < finalMillis) {
|
||||
// Exclude transitions after finalMillis
|
||||
times[nTimes++] = tt;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (nTimes > 0) {
|
||||
// Create a TimeArrayTimeZoneRule
|
||||
raw = rawOffset(typeIdx) * U_MILLIS_PER_SECOND;
|
||||
dst = dstOffset(typeIdx) * U_MILLIS_PER_SECOND;
|
||||
if (historicRules == NULL) {
|
||||
historicRuleCount = typeCount;
|
||||
historicRules = (TimeArrayTimeZoneRule**)uprv_malloc(sizeof(TimeArrayTimeZoneRule*)*historicRuleCount);
|
||||
if (historicRules == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
deleteTransitionRules();
|
||||
uprv_free(times);
|
||||
return;
|
||||
}
|
||||
for (int i = 0; i < historicRuleCount; i++) {
|
||||
// Initialize TimeArrayTimeZoneRule pointers as NULL
|
||||
historicRules[i] = NULL;
|
||||
}
|
||||
}
|
||||
historicRules[typeIdx] = new TimeArrayTimeZoneRule((dst == 0 ? stdName : dstName),
|
||||
raw, dst, times, nTimes, DateTimeRule::UTC_TIME);
|
||||
}
|
||||
}
|
||||
uprv_free(times);
|
||||
|
||||
// Create initial transition
|
||||
typeIdx = (int16_t)typeData[firstTZTransitionIdx];
|
||||
firstTZTransition = new TimeZoneTransition(((UDate)transitionTimes[firstTZTransitionIdx]) * U_MILLIS_PER_SECOND,
|
||||
initialRule->clone(), historicRules[typeIdx]->clone());
|
||||
}
|
||||
}
|
||||
if (initialRule == NULL) {
|
||||
// No historic transitions
|
||||
raw = rawOffset(0) * U_MILLIS_PER_SECOND;
|
||||
dst = dstOffset(0) * U_MILLIS_PER_SECOND;
|
||||
initialRule = new InitialTimeZoneRule((dst == 0 ? stdName : dstName), raw, dst);
|
||||
}
|
||||
if (finalZone != NULL) {
|
||||
// Get the first occurence of final rule starts
|
||||
UDate startTime = (UDate)finalMillis;
|
||||
TimeZoneRule *firstFinalRule = NULL;
|
||||
if (finalZone->useDaylightTime()) {
|
||||
/*
|
||||
* Note: When an OlsonTimeZone is constructed, we should set the final year
|
||||
* as the start year of finalZone. However, the bounday condition used for
|
||||
* getting offset from finalZone has some problems. So setting the start year
|
||||
* in the finalZone will cause a problem. For now, we do not set the valid
|
||||
* start year when the construction time and create a clone and set the
|
||||
* start year when extracting rules.
|
||||
*/
|
||||
finalZoneWithStartYear = (SimpleTimeZone*)finalZone->clone();
|
||||
// finalYear is 1 year before the actual final year.
|
||||
// See the comment in the construction method.
|
||||
finalZoneWithStartYear->setStartYear(finalYear + 1);
|
||||
|
||||
TimeZoneTransition tzt;
|
||||
finalZoneWithStartYear->getNextTransition(startTime, false, tzt);
|
||||
firstFinalRule = tzt.getTo()->clone();
|
||||
startTime = tzt.getTime();
|
||||
} else {
|
||||
finalZoneWithStartYear = (SimpleTimeZone*)finalZone->clone();
|
||||
finalZone->getID(tzid);
|
||||
firstFinalRule = new TimeArrayTimeZoneRule(tzid,
|
||||
finalZone->getRawOffset(), 0, &startTime, 1, DateTimeRule::UTC_TIME);
|
||||
}
|
||||
TimeZoneRule *prevRule = NULL;
|
||||
if (transitionCount > 0) {
|
||||
prevRule = historicRules[typeData[transitionCount - 1]];
|
||||
}
|
||||
if (prevRule == NULL) {
|
||||
// No historic transitions, but only finalZone available
|
||||
prevRule = initialRule;
|
||||
}
|
||||
firstFinalTZTransition = new TimeZoneTransition();
|
||||
firstFinalTZTransition->setTime(startTime);
|
||||
firstFinalTZTransition->adoptFrom(prevRule->clone());
|
||||
firstFinalTZTransition->adoptTo(firstFinalRule);
|
||||
}
|
||||
transitionRulesInitialized = TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
OlsonTimeZone::getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (finalZone != NULL) {
|
||||
if (inclusive && base == firstFinalTZTransition->getTime()) {
|
||||
result = *firstFinalTZTransition;
|
||||
return TRUE;
|
||||
} else if (base >= firstFinalTZTransition->getTime()) {
|
||||
if (finalZone->useDaylightTime()) {
|
||||
//return finalZone->getNextTransition(base, inclusive, result);
|
||||
return finalZoneWithStartYear->getNextTransition(base, inclusive, result);
|
||||
} else {
|
||||
// No more transitions
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (historicRules != NULL) {
|
||||
// Find a historical transition
|
||||
int16_t ttidx = transitionCount - 1;
|
||||
for (; ttidx >= firstTZTransitionIdx; ttidx--) {
|
||||
UDate t = ((UDate)transitionTimes[ttidx]) * U_MILLIS_PER_SECOND;
|
||||
if (base > t || (!inclusive && base == t)) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (ttidx == transitionCount - 1) {
|
||||
if (firstFinalTZTransition != NULL) {
|
||||
result = *firstFinalTZTransition;
|
||||
return TRUE;
|
||||
} else {
|
||||
return FALSE;
|
||||
}
|
||||
} else if (ttidx < firstTZTransitionIdx) {
|
||||
result = *firstTZTransition;
|
||||
return TRUE;
|
||||
} else {
|
||||
// Create a TimeZoneTransition
|
||||
TimeZoneRule *to = historicRules[typeData[ttidx + 1]];
|
||||
TimeZoneRule *from = historicRules[typeData[ttidx]];
|
||||
UDate startTime = ((UDate)transitionTimes[ttidx+1]) * U_MILLIS_PER_SECOND;
|
||||
|
||||
// The transitions loaded from zoneinfo.res may contain non-transition data
|
||||
UnicodeString fromName, toName;
|
||||
from->getName(fromName);
|
||||
to->getName(toName);
|
||||
if (fromName == toName && from->getRawOffset() == to->getRawOffset()
|
||||
&& from->getDSTSavings() == to->getDSTSavings()) {
|
||||
return getNextTransition(startTime, false, result);
|
||||
}
|
||||
result.setTime(startTime);
|
||||
result.adoptFrom(from->clone());
|
||||
result.adoptTo(to->clone());
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
OlsonTimeZone::getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (finalZone != NULL) {
|
||||
if (inclusive && base == firstFinalTZTransition->getTime()) {
|
||||
result = *firstFinalTZTransition;
|
||||
return TRUE;
|
||||
} else if (base > firstFinalTZTransition->getTime()) {
|
||||
if (finalZone->useDaylightTime()) {
|
||||
//return finalZone->getPreviousTransition(base, inclusive, result);
|
||||
return finalZoneWithStartYear->getPreviousTransition(base, inclusive, result);
|
||||
} else {
|
||||
result = *firstFinalTZTransition;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (historicRules != NULL) {
|
||||
// Find a historical transition
|
||||
int16_t ttidx = transitionCount - 1;
|
||||
for (; ttidx >= firstTZTransitionIdx; ttidx--) {
|
||||
UDate t = ((UDate)transitionTimes[ttidx]) * U_MILLIS_PER_SECOND;
|
||||
if (base > t || (inclusive && base == t)) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (ttidx < firstTZTransitionIdx) {
|
||||
// No more transitions
|
||||
return FALSE;
|
||||
} else if (ttidx == firstTZTransitionIdx) {
|
||||
result = *firstTZTransition;
|
||||
return TRUE;
|
||||
} else {
|
||||
// Create a TimeZoneTransition
|
||||
TimeZoneRule *to = historicRules[typeData[ttidx]];
|
||||
TimeZoneRule *from = historicRules[typeData[ttidx-1]];
|
||||
UDate startTime = ((UDate)transitionTimes[ttidx]) * U_MILLIS_PER_SECOND;
|
||||
|
||||
// The transitions loaded from zoneinfo.res may contain non-transition data
|
||||
UnicodeString fromName, toName;
|
||||
from->getName(fromName);
|
||||
to->getName(toName);
|
||||
if (fromName == toName && from->getRawOffset() == to->getRawOffset()
|
||||
&& from->getDSTSavings() == to->getDSTSavings()) {
|
||||
return getPreviousTransition(startTime, false, result);
|
||||
}
|
||||
result.setTime(startTime);
|
||||
result.adoptFrom(from->clone());
|
||||
result.adoptTo(to->clone());
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
int32_t
|
||||
OlsonTimeZone::countTransitionRules(UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return 0;
|
||||
}
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int32_t count = 0;
|
||||
if (historicRules != NULL) {
|
||||
// historicRules may contain null entries when original zoneinfo data
|
||||
// includes non transition data.
|
||||
for (int32_t i = 0; i < historicRuleCount; i++) {
|
||||
if (historicRules[i] != NULL) {
|
||||
count++;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (finalZone != NULL) {
|
||||
if (finalZone->useDaylightTime()) {
|
||||
count += 2;
|
||||
} else {
|
||||
count++;
|
||||
}
|
||||
}
|
||||
return count;
|
||||
}
|
||||
|
||||
void
|
||||
OlsonTimeZone::getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[],
|
||||
int32_t& trscount,
|
||||
UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Initial rule
|
||||
initial = initialRule;
|
||||
|
||||
// Transition rules
|
||||
int32_t cnt = 0;
|
||||
if (historicRules != NULL && trscount > cnt) {
|
||||
// historicRules may contain null entries when original zoneinfo data
|
||||
// includes non transition data.
|
||||
for (int32_t i = 0; i < historicRuleCount; i++) {
|
||||
if (historicRules[i] != NULL) {
|
||||
trsrules[cnt++] = historicRules[i];
|
||||
if (cnt >= trscount) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (finalZoneWithStartYear != NULL && trscount > cnt) {
|
||||
const InitialTimeZoneRule *tmpini;
|
||||
int32_t tmpcnt = trscount - cnt;
|
||||
finalZoneWithStartYear->getTimeZoneRules(tmpini, &trsrules[cnt], tmpcnt, status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
cnt += tmpcnt;
|
||||
}
|
||||
// Set the result length
|
||||
trscount = cnt;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif // !UCONFIG_NO_FORMATTING
|
||||
|
|
|
@ -15,7 +15,7 @@
|
|||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/timezone.h"
|
||||
#include "unicode/basictz.h"
|
||||
|
||||
struct UResourceBundle;
|
||||
|
||||
|
@ -113,7 +113,7 @@ class SimpleTimeZone;
|
|||
* count, the metadata entry itself is considered a rule resource,
|
||||
* since its key begins with an underscore.
|
||||
*/
|
||||
class OlsonTimeZone: public TimeZone {
|
||||
class OlsonTimeZone: public BasicTimeZone {
|
||||
public:
|
||||
/**
|
||||
* Construct from a resource bundle.
|
||||
|
@ -209,8 +209,65 @@ class OlsonTimeZone: public TimeZone {
|
|||
*/
|
||||
virtual UBool inDaylightTime(UDate date, UErrorCode& ec) const;
|
||||
|
||||
/**
|
||||
* TimeZone API.
|
||||
*/
|
||||
virtual int32_t getDSTSavings() const;
|
||||
|
||||
/**
|
||||
* TimeZone API. Also comare historic transitions.
|
||||
*/
|
||||
virtual UBool hasSameRules(const TimeZone& other) const;
|
||||
|
||||
/**
|
||||
* BasicTimeZone API.
|
||||
* Gets the first time zone transition after the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the first transition after the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
*/
|
||||
virtual UBool getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* BasicTimeZone API.
|
||||
* Gets the most recent time zone transition before the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the most recent transition before the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
*/
|
||||
virtual UBool getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* BasicTimeZone API.
|
||||
* Returns the number of <code>TimeZoneRule</code>s which represents time transitions,
|
||||
* for this time zone, that is, all <code>TimeZoneRule</code>s for this time zone except
|
||||
* <code>InitialTimeZoneRule</code>. The return value range is 0 or any positive value.
|
||||
* @param status Receives error status code.
|
||||
* @return The number of <code>TimeZoneRule</code>s representing time transitions.
|
||||
*/
|
||||
virtual int32_t countTransitionRules(UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the <code>InitialTimeZoneRule</code> and the set of <code>TimeZoneRule</code>
|
||||
* which represent time transitions for this time zone. On successful return,
|
||||
* the argument initial points to non-NULL <code>InitialTimeZoneRule</code> and
|
||||
* the array trsrules is filled with 0 or multiple <code>TimeZoneRule</code>
|
||||
* instances up to the size specified by trscount. The results are referencing the
|
||||
* rule instance held by this time zone instance. Therefore, after this time zone
|
||||
* is destructed, they are no longer available.
|
||||
* @param initial Receives the initial timezone rule
|
||||
* @param trsrules Receives the timezone transition rules
|
||||
* @param trscount On input, specify the size of the array 'transitions' receiving
|
||||
* the timezone transition rules. On output, actual number of
|
||||
* rules filled in the array will be set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[], int32_t& trscount, UErrorCode& status) /*const*/;
|
||||
|
||||
private:
|
||||
/**
|
||||
* Default constructor. Creates a time zone with an empty ID and
|
||||
|
@ -278,6 +335,19 @@ private:
|
|||
*/
|
||||
SimpleTimeZone *finalZone; // owned, may be NULL
|
||||
|
||||
/* BasicTimeZone support */
|
||||
void clearTransitionRules(void);
|
||||
void deleteTransitionRules(void);
|
||||
void initTransitionRules(UErrorCode& status);
|
||||
|
||||
InitialTimeZoneRule *initialRule;
|
||||
TimeZoneTransition *firstTZTransition;
|
||||
int16_t firstTZTransitionIdx;
|
||||
TimeZoneTransition *firstFinalTZTransition;
|
||||
TimeArrayTimeZoneRule **historicRules;
|
||||
int16_t historicRuleCount;
|
||||
SimpleTimeZone *finalZoneWithStartYear; // hack
|
||||
UBool transitionRulesInitialized;
|
||||
};
|
||||
|
||||
inline int32_t
|
||||
|
|
827
icu4c/source/i18n/rbtz.cpp
Normal file
827
icu4c/source/i18n/rbtz.cpp
Normal file
|
@ -0,0 +1,827 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/rbtz.h"
|
||||
#include "unicode/gregocal.h"
|
||||
#include "uvector.h"
|
||||
#include "gregoimp.h"
|
||||
#include "cmemory.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
/**
|
||||
* A struct representing a time zone transition
|
||||
*/
|
||||
struct Transition {
|
||||
UDate time;
|
||||
TimeZoneRule* from;
|
||||
TimeZoneRule* to;
|
||||
};
|
||||
|
||||
static UBool compareRules(UVector* rules1, UVector* rules2) {
|
||||
if (rules1 == NULL && rules2 == NULL) {
|
||||
return TRUE;
|
||||
} else if (rules1 == NULL || rules2 == NULL) {
|
||||
return FALSE;
|
||||
}
|
||||
int32_t size = rules1->size();
|
||||
if (size != rules2->size()) {
|
||||
return FALSE;
|
||||
}
|
||||
for (int32_t i = 0; i < size; i++) {
|
||||
TimeZoneRule *r1 = (TimeZoneRule*)rules1->elementAt(i);
|
||||
TimeZoneRule *r2 = (TimeZoneRule*)rules2->elementAt(i);
|
||||
if (r1 != r2) {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static UDate getTransitionTime(Transition* transition, UBool local) {
|
||||
UDate time = transition->time;
|
||||
if (local) {
|
||||
time += transition->from->getRawOffset() + transition->from->getDSTSavings();
|
||||
}
|
||||
return time;
|
||||
}
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(RuleBasedTimeZone)
|
||||
|
||||
RuleBasedTimeZone::RuleBasedTimeZone(const UnicodeString& id, InitialTimeZoneRule* initialRule)
|
||||
: BasicTimeZone(id), fInitialRule(initialRule), fHistoricRules(NULL), fFinalRules(NULL),
|
||||
fHistoricTransitions(NULL), fUpToDate(FALSE) {
|
||||
}
|
||||
|
||||
RuleBasedTimeZone::RuleBasedTimeZone(const RuleBasedTimeZone& source)
|
||||
: BasicTimeZone(source), fInitialRule(source.fInitialRule->clone()),
|
||||
fHistoricRules(NULL), fFinalRules(NULL), fHistoricTransitions(NULL), fUpToDate(FALSE) {
|
||||
}
|
||||
|
||||
RuleBasedTimeZone::~RuleBasedTimeZone() {
|
||||
deleteTransitions();
|
||||
deleteRules();
|
||||
}
|
||||
|
||||
RuleBasedTimeZone&
|
||||
RuleBasedTimeZone::operator=(const RuleBasedTimeZone& right) {
|
||||
if (*this != right) {
|
||||
BasicTimeZone::operator=(right);
|
||||
deleteRules();
|
||||
fInitialRule = right.fInitialRule->clone();
|
||||
fHistoricRules = copyRules(right.fHistoricRules);
|
||||
fFinalRules = copyRules(right.fFinalRules);
|
||||
deleteTransitions();
|
||||
fUpToDate = FALSE;
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::operator==(const RuleBasedTimeZone& that) const {
|
||||
if (this == &that) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != that.getDynamicClassID()
|
||||
|| BasicTimeZone::operator==(that) == FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
if (*fInitialRule != *(that.fInitialRule)) {
|
||||
return FALSE;
|
||||
}
|
||||
if (compareRules(fHistoricRules, that.fHistoricRules)
|
||||
&& compareRules(fFinalRules, that.fFinalRules)) {
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::operator!=(const RuleBasedTimeZone& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::addTransitionRule(TimeZoneRule* rule, UErrorCode& status) {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
if (rule->getDynamicClassID() == AnnualTimeZoneRule::getStaticClassID()
|
||||
&& ((AnnualTimeZoneRule*)rule)->getEndYear() == AnnualTimeZoneRule::MAX_YEAR) {
|
||||
// A final rule
|
||||
if (fFinalRules == NULL) {
|
||||
fFinalRules = new UVector(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
} else if (fFinalRules->size() >= 2) {
|
||||
// Cannot handle more than two final rules
|
||||
status = U_INVALID_STATE_ERROR;
|
||||
return;
|
||||
}
|
||||
fFinalRules->addElement((void*)rule, status);
|
||||
} else {
|
||||
// Non-final rule
|
||||
if (fHistoricRules == NULL) {
|
||||
fHistoricRules = new UVector(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
fHistoricRules->addElement((void*)rule, status);
|
||||
}
|
||||
// Mark dirty, so transitions are recalculated at next complete() call
|
||||
fUpToDate = FALSE;
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::complete(UErrorCode& status) {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
if (fUpToDate) {
|
||||
return;
|
||||
}
|
||||
// Make sure either no final rules or a pair of AnnualTimeZoneRules
|
||||
// are available.
|
||||
if (fFinalRules != NULL && fFinalRules->size() != 2) {
|
||||
status = U_INVALID_STATE_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
UBool *done = NULL;
|
||||
// Create a TimezoneTransition and add to the list
|
||||
if (fHistoricRules != NULL || fFinalRules != NULL) {
|
||||
TimeZoneRule *curRule = fInitialRule;
|
||||
UDate lastTransitionTime = MIN_MILLIS;
|
||||
|
||||
// Build the transition array which represents historical time zone
|
||||
// transitions.
|
||||
if (fHistoricRules != NULL && fHistoricRules->size() > 0) {
|
||||
int32_t i;
|
||||
int32_t historicCount = fHistoricRules->size();
|
||||
done = (UBool*)uprv_malloc(sizeof(UBool) * historicCount);
|
||||
if (done == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto cleanup;
|
||||
}
|
||||
for (i = 0; i < historicCount; i++) {
|
||||
done[i] = FALSE;
|
||||
}
|
||||
while (true) {
|
||||
int32_t curStdOffset = curRule->getRawOffset();
|
||||
int32_t curDstSavings = curRule->getDSTSavings();
|
||||
UDate nextTransitionTime = MAX_MILLIS;
|
||||
TimeZoneRule *nextRule = NULL;
|
||||
TimeZoneRule *r = NULL;
|
||||
UBool avail;
|
||||
UDate tt;
|
||||
UnicodeString curName, name;
|
||||
curRule->getName(curName);
|
||||
|
||||
for (i = 0; i < historicCount; i++) {
|
||||
if (done[i]) {
|
||||
continue;
|
||||
}
|
||||
r = (TimeZoneRule*)fHistoricRules->elementAt(i);
|
||||
avail = r->getNextStart(lastTransitionTime, curStdOffset, curDstSavings, false, tt);
|
||||
if (!avail) {
|
||||
// No more transitions from this rule - skip this rule next time
|
||||
done[i] = TRUE;
|
||||
} else {
|
||||
r->getName(name);
|
||||
if (*r == *curRule ||
|
||||
(name == curName && r->getRawOffset() == curRule->getRawOffset()
|
||||
&& r->getDSTSavings() == curRule->getDSTSavings())) {
|
||||
continue;
|
||||
}
|
||||
if (tt < nextTransitionTime) {
|
||||
nextTransitionTime = tt;
|
||||
nextRule = r;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (nextRule == NULL) {
|
||||
// Check if all historic rules are done
|
||||
UBool bDoneAll = TRUE;
|
||||
for (int32_t j = 0; j < historicCount; j++) {
|
||||
if (!done[j]) {
|
||||
bDoneAll = FALSE;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (bDoneAll) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (fFinalRules != NULL) {
|
||||
// Check if one of final rules has earlier transition date
|
||||
for (i = 0; i < 2 /* fFinalRules->size() */; i++) {
|
||||
TimeZoneRule *fr = (TimeZoneRule*)fFinalRules->elementAt(i);
|
||||
if (*fr == *curRule) {
|
||||
continue;
|
||||
}
|
||||
r = (TimeZoneRule*)fFinalRules->elementAt(i);
|
||||
avail = r->getNextStart(lastTransitionTime, curStdOffset, curDstSavings, false, tt);
|
||||
if (avail) {
|
||||
if (tt < nextTransitionTime) {
|
||||
nextTransitionTime = tt;
|
||||
nextRule = r;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (nextRule == NULL) {
|
||||
// Nothing more
|
||||
break;
|
||||
}
|
||||
|
||||
if (fHistoricTransitions == NULL) {
|
||||
fHistoricTransitions = new UVector(status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
Transition *trst = (Transition*)uprv_malloc(sizeof(Transition));
|
||||
if (trst == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto cleanup;
|
||||
}
|
||||
trst->time = nextTransitionTime;
|
||||
trst->from = curRule;
|
||||
trst->to = nextRule;
|
||||
fHistoricTransitions->addElement(trst, status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto cleanup;
|
||||
}
|
||||
lastTransitionTime = nextTransitionTime;
|
||||
curRule = nextRule;
|
||||
}
|
||||
}
|
||||
if (fFinalRules != NULL) {
|
||||
if (fHistoricTransitions == NULL) {
|
||||
fHistoricTransitions = new UVector(status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
// Append the first transition for each
|
||||
TimeZoneRule *rule0 = (TimeZoneRule*)fFinalRules->elementAt(0);
|
||||
TimeZoneRule *rule1 = (TimeZoneRule*)fFinalRules->elementAt(1);
|
||||
UDate tt0, tt1;
|
||||
UBool avail0 = rule0->getNextStart(lastTransitionTime, curRule->getRawOffset(), curRule->getDSTSavings(), false, tt0);
|
||||
UBool avail1 = rule1->getNextStart(lastTransitionTime, curRule->getRawOffset(), curRule->getDSTSavings(), false, tt1);
|
||||
if (!avail0 || !avail1) {
|
||||
// Should not happen, because both rules are permanent
|
||||
status = U_INVALID_STATE_ERROR;
|
||||
goto cleanup;
|
||||
}
|
||||
Transition *final0 = (Transition*)uprv_malloc(sizeof(Transition));
|
||||
if (final0 == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto cleanup;
|
||||
}
|
||||
Transition *final1 = (Transition*)uprv_malloc(sizeof(Transition));
|
||||
if (final1 == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
goto cleanup;
|
||||
}
|
||||
if (tt0 < tt1) {
|
||||
final0->time = tt0;
|
||||
final0->from = curRule;
|
||||
final0->to = rule0;
|
||||
rule1->getNextStart(tt0, rule0->getRawOffset(), rule0->getDSTSavings(), false, final1->time);
|
||||
final1->from = rule0;
|
||||
final1->to = rule1;
|
||||
} else {
|
||||
final0->time = tt1;
|
||||
final0->from = curRule;
|
||||
final0->to = rule1;
|
||||
rule0->getNextStart(tt1, rule1->getRawOffset(), rule1->getDSTSavings(), false, final1->time);
|
||||
final1->from = rule1;
|
||||
final1->to = rule0;
|
||||
}
|
||||
fHistoricTransitions->addElement(final0, status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto cleanup;
|
||||
}
|
||||
fHistoricTransitions->addElement(final1, status);
|
||||
if (U_FAILURE(status)) {
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
}
|
||||
fUpToDate = TRUE;
|
||||
return;
|
||||
|
||||
cleanup:
|
||||
deleteTransitions();
|
||||
if (done != NULL) {
|
||||
uprv_free(done);
|
||||
}
|
||||
fUpToDate = FALSE;
|
||||
}
|
||||
|
||||
TimeZone*
|
||||
RuleBasedTimeZone::clone(void) const {
|
||||
return new RuleBasedTimeZone(*this);
|
||||
}
|
||||
|
||||
int32_t
|
||||
RuleBasedTimeZone::getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis, UErrorCode& status) const {
|
||||
if (U_FAILURE(status)) {
|
||||
return 0;
|
||||
}
|
||||
if (month < UCAL_JANUARY || month > UCAL_DECEMBER) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return 0;
|
||||
} else {
|
||||
return getOffset(era, year, month, day, dayOfWeek, millis,
|
||||
Grego::monthLength(year, month), status);
|
||||
}
|
||||
}
|
||||
|
||||
int32_t
|
||||
RuleBasedTimeZone::getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis,
|
||||
int32_t monthLength, UErrorCode& status) const {
|
||||
// dayOfWeek and monthLength are unused
|
||||
if (U_FAILURE(status)) {
|
||||
return 0;
|
||||
}
|
||||
if (era == GregorianCalendar::BC) {
|
||||
// Convert to extended year
|
||||
year = 1 - year;
|
||||
}
|
||||
int32_t rawOffset, dstOffset;
|
||||
UDate time = (UDate)Grego::fieldsToDay(year, month, day) * U_MILLIS_PER_DAY + millis;
|
||||
getOffset(time, true, rawOffset, dstOffset, status);
|
||||
if (U_FAILURE(status)) {
|
||||
return 0;
|
||||
}
|
||||
return (rawOffset + dstOffset);
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::getOffset(UDate date, UBool local, int32_t& rawOffset,
|
||||
int32_t& dstOffset, UErrorCode& status) const {
|
||||
rawOffset = 0;
|
||||
dstOffset = 0;
|
||||
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
if (!fUpToDate) {
|
||||
// Transitions are not yet resolved. We cannot do it here
|
||||
// because this method is const. Thus, do nothing and return
|
||||
// error status.
|
||||
status = U_INVALID_STATE_ERROR;
|
||||
return;
|
||||
}
|
||||
const TimeZoneRule *rule = NULL;
|
||||
if (fHistoricTransitions == NULL) {
|
||||
rule = fInitialRule;
|
||||
} else {
|
||||
UDate tstart = getTransitionTime((Transition*)fHistoricTransitions->elementAt(0), local);
|
||||
if (date < tstart) {
|
||||
rule = fInitialRule;
|
||||
} else {
|
||||
int32_t idx = fHistoricTransitions->size() - 1;
|
||||
UDate tend = getTransitionTime((Transition*)fHistoricTransitions->elementAt(idx), local);
|
||||
if (date > tend) {
|
||||
if (fFinalRules != NULL) {
|
||||
rule = findRuleInFinal(date, local);
|
||||
} else {
|
||||
// no final rule, use the last rule
|
||||
rule = ((Transition*)fHistoricTransitions->elementAt(idx))->to;
|
||||
}
|
||||
} else {
|
||||
// Find a historical transition
|
||||
while (idx >= 0) {
|
||||
if (date >= getTransitionTime((Transition*)fHistoricTransitions->elementAt(idx), local)) {
|
||||
break;
|
||||
}
|
||||
idx--;
|
||||
}
|
||||
rule = ((Transition*)fHistoricTransitions->elementAt(idx))->to;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (rule != NULL) {
|
||||
rawOffset = rule->getRawOffset();
|
||||
dstOffset = rule->getDSTSavings();
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::setRawOffset(int32_t offsetMillis) {
|
||||
// We don't support this operation at this moment.
|
||||
// Nothing to do!
|
||||
}
|
||||
|
||||
int32_t
|
||||
RuleBasedTimeZone::getRawOffset(void) const {
|
||||
// Note: This implementation returns standard GMT offset
|
||||
// as of current time.
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
int32_t raw, dst;
|
||||
getOffset(uprv_getUTCtime() * U_MILLIS_PER_SECOND,
|
||||
FALSE, raw, dst, status);
|
||||
return raw;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::useDaylightTime(void) const {
|
||||
// Note: This implementation returns true when
|
||||
// daylight saving time is used as of now or
|
||||
// after the next transition.
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
UDate now = uprv_getUTCtime() * U_MILLIS_PER_SECOND;
|
||||
int32_t raw, dst;
|
||||
getOffset(now, FALSE, raw, dst, status);
|
||||
if (dst != 0) {
|
||||
return TRUE;
|
||||
}
|
||||
// If DST is not used now, check if DST is used after the next transition
|
||||
UDate time;
|
||||
TimeZoneRule *from, *to;
|
||||
UBool avail = findNext(now, FALSE, time, from, to);
|
||||
if (avail && to->getDSTSavings() != 0) {
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::inDaylightTime(UDate date, UErrorCode& status) const {
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
int32_t raw, dst;
|
||||
getOffset(date, FALSE, raw, dst, status);
|
||||
if (dst != 0) {
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::hasSameRules(const TimeZone& other) const {
|
||||
if (this == &other) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != other.getDynamicClassID()) {
|
||||
return FALSE;
|
||||
}
|
||||
const RuleBasedTimeZone& that = (const RuleBasedTimeZone&)other;
|
||||
if (*fInitialRule != *(that.fInitialRule)) {
|
||||
return FALSE;
|
||||
}
|
||||
if (compareRules(fHistoricRules, that.fHistoricRules)
|
||||
&& compareRules(fFinalRules, that.fFinalRules)) {
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
complete(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
UDate transitionTime;
|
||||
TimeZoneRule *fromRule, *toRule;
|
||||
UBool found = findNext(base, inclusive, transitionTime, fromRule, toRule);
|
||||
if (found) {
|
||||
result.setTime(transitionTime);
|
||||
result.setFrom((const TimeZoneRule&)*fromRule);
|
||||
result.setTo((const TimeZoneRule&)*toRule);
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
complete(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
UDate transitionTime;
|
||||
TimeZoneRule *fromRule, *toRule;
|
||||
UBool found = findPrev(base, inclusive, transitionTime, fromRule, toRule);
|
||||
if (found) {
|
||||
result.setTime(transitionTime);
|
||||
result.setFrom((const TimeZoneRule&)*fromRule);
|
||||
result.setTo((const TimeZoneRule&)*toRule);
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
int32_t
|
||||
RuleBasedTimeZone::countTransitionRules(UErrorCode& status) /*const*/ {
|
||||
int32_t count = 0;
|
||||
if (fHistoricRules != NULL) {
|
||||
count += fHistoricRules->size();
|
||||
}
|
||||
if (fFinalRules != NULL) {
|
||||
count += fFinalRules->size();
|
||||
}
|
||||
return count;
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[],
|
||||
int32_t& trscount,
|
||||
UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
// Initial rule
|
||||
initial = fInitialRule;
|
||||
|
||||
// Transition rules
|
||||
int32_t cnt = 0;
|
||||
int32_t idx;
|
||||
if (fHistoricRules != NULL && cnt < trscount) {
|
||||
int32_t historicCount = fHistoricRules->size();
|
||||
idx = 0;
|
||||
while (cnt < trscount && idx < historicCount) {
|
||||
trsrules[cnt++] = (const TimeZoneRule*)fHistoricRules->elementAt(idx);
|
||||
}
|
||||
}
|
||||
if (fFinalRules != NULL && cnt < trscount) {
|
||||
int32_t finalCount = fFinalRules->size();
|
||||
idx = 0;
|
||||
while (cnt < trscount && idx < finalCount) {
|
||||
trsrules[cnt++] = (const TimeZoneRule*)fFinalRules->elementAt(idx);
|
||||
}
|
||||
}
|
||||
// Set the result length
|
||||
trscount = cnt;
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::deleteRules(void) {
|
||||
delete fInitialRule;
|
||||
fInitialRule = NULL;
|
||||
if (fHistoricRules != NULL) {
|
||||
while (!fHistoricRules->isEmpty()) {
|
||||
delete (TimeZoneRule*)(fHistoricRules->orphanElementAt(0));
|
||||
}
|
||||
delete fHistoricRules;
|
||||
fHistoricRules = NULL;
|
||||
}
|
||||
if (fFinalRules != NULL) {
|
||||
while (!fFinalRules->isEmpty()) {
|
||||
delete (AnnualTimeZoneRule*)(fFinalRules->orphanElementAt(0));
|
||||
}
|
||||
delete fFinalRules;
|
||||
fFinalRules = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
RuleBasedTimeZone::deleteTransitions(void) {
|
||||
if (fHistoricTransitions != NULL) {
|
||||
while (!fHistoricTransitions->isEmpty()) {
|
||||
Transition *trs = (Transition*)fHistoricTransitions->orphanElementAt(0);
|
||||
uprv_free(trs);
|
||||
}
|
||||
delete fHistoricTransitions;
|
||||
}
|
||||
fHistoricTransitions = NULL;
|
||||
}
|
||||
|
||||
UVector*
|
||||
RuleBasedTimeZone::copyRules(UVector* source) {
|
||||
if (source == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
UErrorCode ec = U_ZERO_ERROR;
|
||||
int32_t size = source->size();
|
||||
UVector *rules = new UVector(size, ec);
|
||||
if (U_FAILURE(ec)) {
|
||||
return NULL;
|
||||
}
|
||||
int32_t i;
|
||||
for (i = 0; i < size; i++) {
|
||||
rules->addElement(((TimeZoneRule*)source->elementAt(i))->clone(), ec);
|
||||
if (U_FAILURE(ec)) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (U_FAILURE(ec)) {
|
||||
// In case of error, clean up
|
||||
for (i = 0; i < rules->size(); i++) {
|
||||
TimeZoneRule *rule = (TimeZoneRule*)rules->orphanElementAt(i);
|
||||
delete rule;
|
||||
}
|
||||
delete rules;
|
||||
return NULL;
|
||||
}
|
||||
return rules;
|
||||
}
|
||||
|
||||
TimeZoneRule*
|
||||
RuleBasedTimeZone::findRuleInFinal(UDate date, UBool local) const {
|
||||
if (fFinalRules == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule* fr0 = (AnnualTimeZoneRule*)fFinalRules->elementAt(0);
|
||||
AnnualTimeZoneRule* fr1 = (AnnualTimeZoneRule*)fFinalRules->elementAt(1);
|
||||
if (fr0 == NULL || fr1 == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
UDate start0, start1;
|
||||
UDate base;
|
||||
|
||||
base = local ? date - fr1->getRawOffset() - fr1->getDSTSavings() : date;
|
||||
UBool avail0 = fr0->getPreviousStart(base, fr1->getRawOffset(), fr1->getDSTSavings(), true, start0);
|
||||
|
||||
base = local ? date - fr0->getRawOffset() - fr0->getDSTSavings() : date;
|
||||
UBool avail1 = fr1->getPreviousStart(base, fr0->getRawOffset(), fr0->getDSTSavings(), true, start1);
|
||||
|
||||
if (avail0 && (!avail1 || start0 > start1)) {
|
||||
return fr0;
|
||||
} else if (avail1) {
|
||||
return fr1;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::findNext(UDate base, UBool inclusive, UDate& transitionTime,
|
||||
TimeZoneRule*& fromRule, TimeZoneRule*& toRule) const {
|
||||
if (fHistoricTransitions == NULL) {
|
||||
return FALSE;
|
||||
}
|
||||
UBool isFinal = FALSE;
|
||||
UBool found = FALSE;
|
||||
Transition result;
|
||||
Transition *tzt = (Transition*)fHistoricTransitions->elementAt(0);
|
||||
UDate tt = getTransitionTime(tzt, FALSE);
|
||||
if (tt > base || (inclusive && tt == base)) {
|
||||
result = *tzt;
|
||||
found = TRUE;
|
||||
} else {
|
||||
int32_t idx = fHistoricTransitions->size() - 1;
|
||||
tzt = (Transition*)fHistoricTransitions->elementAt(idx);
|
||||
tt = getTransitionTime(tzt, FALSE);
|
||||
if (inclusive && tt == base) {
|
||||
result = *tzt;
|
||||
found = TRUE;
|
||||
} else if (tt <= base) {
|
||||
if (fFinalRules != NULL) {
|
||||
// Find a transion time with finalRules
|
||||
TimeZoneRule *r0 = (TimeZoneRule*)fFinalRules->elementAt(0);
|
||||
TimeZoneRule *r1 = (TimeZoneRule*)fFinalRules->elementAt(1);
|
||||
UDate start0, start1;
|
||||
UBool avail0 = r0->getNextStart(base, r1->getRawOffset(), r1->getDSTSavings(), inclusive, start0);
|
||||
UBool avail1 = r1->getNextStart(base, r0->getRawOffset(), r0->getDSTSavings(), inclusive, start1);
|
||||
// avail0/avail1 should be always TRUE
|
||||
if (start0 < start1) {
|
||||
result.time = start0;
|
||||
result.from = r1;
|
||||
result.to = r0;
|
||||
} else {
|
||||
result.time = start1;
|
||||
result.from = r0;
|
||||
result.to = r1;
|
||||
}
|
||||
isFinal = TRUE;
|
||||
found = TRUE;
|
||||
}
|
||||
} else {
|
||||
// Find a transition within the historic transitions
|
||||
idx--;
|
||||
Transition *prev = tzt;
|
||||
while (idx > 0) {
|
||||
tzt = (Transition*)fHistoricTransitions->elementAt(idx);
|
||||
tt = getTransitionTime(tzt, FALSE);
|
||||
if (tt < base || (!inclusive && tt == base)) {
|
||||
break;
|
||||
}
|
||||
idx--;
|
||||
prev = tzt;
|
||||
}
|
||||
result.time = prev->time;
|
||||
result.from = prev->from;
|
||||
result.to = prev->to;
|
||||
found = TRUE;
|
||||
}
|
||||
}
|
||||
if (found) {
|
||||
// For now, this implementation ignore transitions with only zone name changes.
|
||||
if (result.from->getRawOffset() == result.to->getRawOffset()
|
||||
&& result.from->getDSTSavings() == result.to->getDSTSavings()) {
|
||||
// No offset changes. Try next one if not final
|
||||
return findNext(result.time, FALSE /* always exclusive */,
|
||||
transitionTime, fromRule, toRule);
|
||||
}
|
||||
transitionTime = result.time;
|
||||
fromRule = result.from;
|
||||
toRule = result.to;
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
RuleBasedTimeZone::findPrev(UDate base, UBool inclusive, UDate& transitionTime,
|
||||
TimeZoneRule*& fromRule, TimeZoneRule*& toRule) const {
|
||||
if (fHistoricTransitions == NULL) {
|
||||
return FALSE;
|
||||
}
|
||||
UBool found = FALSE;
|
||||
Transition result;
|
||||
Transition *tzt = (Transition*)fHistoricTransitions->elementAt(0);
|
||||
UDate tt = getTransitionTime(tzt, FALSE);
|
||||
if (inclusive && tt == base) {
|
||||
result = *tzt;
|
||||
found = TRUE;
|
||||
} else if (tt < base) {
|
||||
int32_t idx = fHistoricTransitions->size() - 1;
|
||||
tzt = (Transition*)fHistoricTransitions->elementAt(idx);
|
||||
tt = getTransitionTime(tzt, FALSE);
|
||||
if (inclusive && tt == base) {
|
||||
result = *tzt;
|
||||
found = TRUE;
|
||||
} else if (tt < base) {
|
||||
if (fFinalRules != NULL) {
|
||||
// Find a transion time with finalRules
|
||||
TimeZoneRule *r0 = (TimeZoneRule*)fFinalRules->elementAt(0);
|
||||
TimeZoneRule *r1 = (TimeZoneRule*)fFinalRules->elementAt(1);
|
||||
UDate start0, start1;
|
||||
UBool avail0 = r0->getPreviousStart(base, r1->getRawOffset(), r1->getDSTSavings(), inclusive, start0);
|
||||
UBool avail1 = r1->getPreviousStart(base, r0->getRawOffset(), r0->getDSTSavings(), inclusive, start1);
|
||||
// avail0/avail1 should be always TRUE
|
||||
if (start0 > start1) {
|
||||
result.time = start0;
|
||||
result.from = r1;
|
||||
result.to = r0;
|
||||
} else {
|
||||
result.time = start1;
|
||||
result.from = r0;
|
||||
result.to = r1;
|
||||
}
|
||||
} else {
|
||||
result = *tzt;
|
||||
}
|
||||
found = TRUE;
|
||||
} else {
|
||||
// Find a transition within the historic transitions
|
||||
idx--;
|
||||
while (idx >= 0) {
|
||||
tzt = (Transition*)fHistoricTransitions->elementAt(idx);
|
||||
tt = getTransitionTime(tzt, FALSE);
|
||||
if (tt < base || (inclusive && tt == base)) {
|
||||
break;
|
||||
}
|
||||
idx--;
|
||||
}
|
||||
result = *tzt;
|
||||
found = TRUE;
|
||||
}
|
||||
}
|
||||
if (found) {
|
||||
// For now, this implementation ignore transitions with only zone name changes.
|
||||
if (result.from->getRawOffset() == result.to->getRawOffset()
|
||||
&& result.from->getDSTSavings() == result.to->getDSTSavings()) {
|
||||
// No offset changes. Try next one if not final
|
||||
return findPrev(result.time, FALSE /* always exclusive */,
|
||||
transitionTime, fromRule, toRule);
|
||||
}
|
||||
transitionTime = result.time;
|
||||
fromRule = result.from;
|
||||
toRule = result.to;
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
//eof
|
||||
|
|
@ -53,7 +53,7 @@ const int8_t SimpleTimeZone::STATICMONTHLENGTH[] = {31,29,31,30,31,30,31,31,30,3
|
|||
|
||||
|
||||
SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID)
|
||||
: TimeZone(ID),
|
||||
: BasicTimeZone(ID),
|
||||
startMonth(0),
|
||||
startDay(0),
|
||||
startDayOfWeek(0),
|
||||
|
@ -71,6 +71,7 @@ SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID)
|
|||
endMode(DOM_MODE),
|
||||
dstSavings(U_MILLIS_PER_HOUR)
|
||||
{
|
||||
clearTransitionRules();
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -81,8 +82,9 @@ SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
|
|||
int8_t savingsEndMonth, int8_t savingsEndDay,
|
||||
int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
|
||||
UErrorCode& status)
|
||||
: TimeZone(ID)
|
||||
: BasicTimeZone(ID)
|
||||
{
|
||||
clearTransitionRules();
|
||||
construct(rawOffsetGMT,
|
||||
savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
|
||||
savingsStartTime, WALL_TIME,
|
||||
|
@ -99,8 +101,9 @@ SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
|
|||
int8_t savingsEndMonth, int8_t savingsEndDay,
|
||||
int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
|
||||
int32_t savingsDST, UErrorCode& status)
|
||||
: TimeZone(ID)
|
||||
: BasicTimeZone(ID)
|
||||
{
|
||||
clearTransitionRules();
|
||||
construct(rawOffsetGMT,
|
||||
savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
|
||||
savingsStartTime, WALL_TIME,
|
||||
|
@ -119,8 +122,9 @@ SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
|
|||
int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
|
||||
TimeMode savingsEndTimeMode,
|
||||
int32_t savingsDST, UErrorCode& status)
|
||||
: TimeZone(ID)
|
||||
: BasicTimeZone(ID)
|
||||
{
|
||||
clearTransitionRules();
|
||||
construct(rawOffsetGMT,
|
||||
savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
|
||||
savingsStartTime, savingsStartTimeMode,
|
||||
|
@ -173,13 +177,14 @@ void SimpleTimeZone::construct(int32_t rawOffsetGMT,
|
|||
|
||||
SimpleTimeZone::~SimpleTimeZone()
|
||||
{
|
||||
deleteTransitionRules();
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
||||
// Called by TimeZone::createDefault(), then clone() inside a Mutex - be careful.
|
||||
SimpleTimeZone::SimpleTimeZone(const SimpleTimeZone &source)
|
||||
: TimeZone(source)
|
||||
: BasicTimeZone(source)
|
||||
{
|
||||
*this = source;
|
||||
}
|
||||
|
@ -209,6 +214,7 @@ SimpleTimeZone::operator=(const SimpleTimeZone &right)
|
|||
startYear = right.startYear;
|
||||
dstSavings = right.dstSavings;
|
||||
useDaylight = right.useDaylight;
|
||||
clearTransitionRules();
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
@ -246,6 +252,7 @@ void
|
|||
SimpleTimeZone::setStartYear(int32_t year)
|
||||
{
|
||||
startYear = year;
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -299,6 +306,7 @@ SimpleTimeZone::setStartRule(int32_t month, int32_t dayOfWeekInMonth, int32_t da
|
|||
startTime = time;
|
||||
startTimeMode = mode;
|
||||
decodeStartRule(status);
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -350,6 +358,7 @@ SimpleTimeZone::setEndRule(int32_t month, int32_t dayOfWeekInMonth, int32_t dayO
|
|||
endTime = time;
|
||||
endTimeMode = mode;
|
||||
decodeEndRule(status);
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -611,6 +620,7 @@ void
|
|||
SimpleTimeZone::setRawOffset(int32_t offsetMillis)
|
||||
{
|
||||
rawOffset = offsetMillis;
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -624,6 +634,7 @@ SimpleTimeZone::setDSTSavings(int32_t millisSavedDuringDST, UErrorCode& status)
|
|||
else {
|
||||
dstSavings = millisSavedDuringDST;
|
||||
}
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
// -------------------------------------
|
||||
|
@ -900,6 +911,210 @@ SimpleTimeZone::decodeEndRule(UErrorCode& status)
|
|||
}
|
||||
}
|
||||
|
||||
UBool
|
||||
SimpleTimeZone::getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
if (startMonth == 0) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UDate firstTransitionTime = firstTransition->getTime();
|
||||
if (base < firstTransitionTime || (inclusive && base == firstTransitionTime)) {
|
||||
result = *firstTransition;
|
||||
}
|
||||
UDate stdDate, dstDate;
|
||||
UBool stdAvail = stdRule->getNextStart(base, dstRule->getRawOffset(), dstRule->getDSTSavings(), inclusive, stdDate);
|
||||
UBool dstAvail = dstRule->getNextStart(base, stdRule->getRawOffset(), stdRule->getDSTSavings(), inclusive, dstDate);
|
||||
if (stdAvail && (!dstAvail || stdDate < dstDate)) {
|
||||
result.setTime(stdDate);
|
||||
result.setFrom((const TimeZoneRule&)*dstRule);
|
||||
result.setTo((const TimeZoneRule&)*stdRule);
|
||||
return TRUE;
|
||||
}
|
||||
if (dstAvail && (!stdAvail || dstDate < stdDate)) {
|
||||
result.setTime(dstDate);
|
||||
result.setFrom((const TimeZoneRule&)*stdRule);
|
||||
result.setTo((const TimeZoneRule&)*dstRule);
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
SimpleTimeZone::getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
|
||||
if (startMonth == 0) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UDate firstTransitionTime = firstTransition->getTime();
|
||||
if (base < firstTransitionTime || (!inclusive && base == firstTransitionTime)) {
|
||||
return FALSE;
|
||||
}
|
||||
UDate stdDate, dstDate;
|
||||
UBool stdAvail = stdRule->getPreviousStart(base, dstRule->getRawOffset(), dstRule->getDSTSavings(), false, stdDate);
|
||||
UBool dstAvail = dstRule->getPreviousStart(base, stdRule->getRawOffset(), stdRule->getDSTSavings(), false, dstDate);
|
||||
if (stdAvail && (!dstAvail || stdDate > dstDate)) {
|
||||
result.setTime(stdDate);
|
||||
result.setFrom((const TimeZoneRule&)*dstRule);
|
||||
result.setTo((const TimeZoneRule&)*stdRule);
|
||||
return TRUE;
|
||||
}
|
||||
if (dstAvail && (!stdAvail || dstDate > stdDate)) {
|
||||
result.setTime(dstDate);
|
||||
result.setFrom((const TimeZoneRule&)*stdRule);
|
||||
result.setTo((const TimeZoneRule&)*dstRule);
|
||||
return TRUE;
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
void
|
||||
SimpleTimeZone::clearTransitionRules(void) {
|
||||
initialRule = NULL;
|
||||
firstTransition = NULL;
|
||||
stdRule = NULL;
|
||||
dstRule = NULL;
|
||||
transitionRulesInitialized = FALSE;
|
||||
}
|
||||
|
||||
void
|
||||
SimpleTimeZone::deleteTransitionRules(void) {
|
||||
if (initialRule != NULL) {
|
||||
delete initialRule;
|
||||
}
|
||||
if (firstTransition != NULL) {
|
||||
delete firstTransition;
|
||||
}
|
||||
if (stdRule != NULL) {
|
||||
delete stdRule;
|
||||
}
|
||||
if (dstRule != NULL) {
|
||||
delete dstRule;
|
||||
}
|
||||
clearTransitionRules();
|
||||
}
|
||||
|
||||
void
|
||||
SimpleTimeZone::initTransitionRules(UErrorCode& status) {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
if (transitionRulesInitialized) {
|
||||
return;
|
||||
}
|
||||
deleteTransitionRules();
|
||||
UnicodeString tzid;
|
||||
getID(tzid);
|
||||
|
||||
if (startMonth != 0) {
|
||||
DateTimeRule* dtRule;
|
||||
DateTimeRule::TimeRuleType timeRuleType;
|
||||
UDate firstStdStart, firstDstStart;
|
||||
|
||||
// Create a TimeZoneRule for daylight saving time
|
||||
timeRuleType = (startTimeMode == STANDARD_TIME) ? DateTimeRule::STANDARD_TIME :
|
||||
((startTimeMode == UTC_TIME) ? DateTimeRule::UTC_TIME : DateTimeRule::WALL_TIME);
|
||||
switch (startMode) {
|
||||
case DOM_MODE:
|
||||
dtRule = new DateTimeRule(startMonth, startDay, startTime, timeRuleType);
|
||||
break;
|
||||
case DOW_IN_MONTH_MODE:
|
||||
dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, startTime, timeRuleType);
|
||||
break;
|
||||
case DOW_GE_DOM_MODE:
|
||||
dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, true, startTime, timeRuleType);
|
||||
break;
|
||||
case DOW_LE_DOM_MODE:
|
||||
dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, false, startTime, timeRuleType);
|
||||
break;
|
||||
}
|
||||
// For now, use ID + "(DST)" as the name
|
||||
dstRule = new AnnualTimeZoneRule(tzid+"(DST)", getRawOffset(), getDSTSavings(),
|
||||
*dtRule, startYear, AnnualTimeZoneRule::MAX_YEAR);
|
||||
|
||||
// Calculate the first DST start time
|
||||
dstRule->getFirstStart(getRawOffset(), 0, firstDstStart);
|
||||
|
||||
// Create a TimeZoneRule for standard time
|
||||
timeRuleType = (endTimeMode == STANDARD_TIME) ? DateTimeRule::STANDARD_TIME :
|
||||
((endTimeMode == UTC_TIME) ? DateTimeRule::UTC_TIME : DateTimeRule::WALL_TIME);
|
||||
switch (endMode) {
|
||||
case DOM_MODE:
|
||||
dtRule = new DateTimeRule(endMonth, endDay, endTime, timeRuleType);
|
||||
break;
|
||||
case DOW_IN_MONTH_MODE:
|
||||
dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, endTime, timeRuleType);
|
||||
break;
|
||||
case DOW_GE_DOM_MODE:
|
||||
dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, true, endTime, timeRuleType);
|
||||
break;
|
||||
case DOW_LE_DOM_MODE:
|
||||
dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, false, endTime, timeRuleType);
|
||||
break;
|
||||
}
|
||||
// For now, use ID + "(STD)" as the name
|
||||
stdRule = new AnnualTimeZoneRule(tzid+"(STD)", getRawOffset(), 0,
|
||||
*dtRule, startYear, AnnualTimeZoneRule::MAX_YEAR);
|
||||
|
||||
// Calculate the first STD start time
|
||||
stdRule->getFirstStart(getRawOffset(), dstRule->getDSTSavings(), firstStdStart);
|
||||
|
||||
// Create a TimeZoneRule for initial time
|
||||
if (firstStdStart < firstDstStart) {
|
||||
initialRule = new InitialTimeZoneRule(tzid+"(DST)", getRawOffset(), dstRule->getDSTSavings());
|
||||
firstTransition = new TimeZoneTransition(firstStdStart, initialRule->clone(), stdRule->clone());
|
||||
} else {
|
||||
initialRule = new InitialTimeZoneRule(tzid+"(STD)", getRawOffset(), 0);
|
||||
firstTransition = new TimeZoneTransition(firstDstStart, initialRule->clone(), dstRule->clone());
|
||||
}
|
||||
|
||||
} else {
|
||||
// Create a TimeZoneRule for initial time
|
||||
initialRule = new InitialTimeZoneRule(tzid, getRawOffset(), 0);
|
||||
}
|
||||
transitionRulesInitialized = true;
|
||||
}
|
||||
|
||||
int32_t
|
||||
SimpleTimeZone::countTransitionRules(UErrorCode& status) /*const*/ {
|
||||
return (startMonth == 0) ? 0 : 2;
|
||||
}
|
||||
|
||||
void
|
||||
SimpleTimeZone::getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[],
|
||||
int32_t& trscount,
|
||||
UErrorCode& status) /*const*/ {
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
initTransitionRules(status);
|
||||
if (U_FAILURE(status)) {
|
||||
return;
|
||||
}
|
||||
int32_t cnt = 0;
|
||||
initial = initialRule;
|
||||
if (trscount > 0) {
|
||||
trsrules[cnt++] = stdRule;
|
||||
}
|
||||
if (trscount > 1) {
|
||||
trsrules[cnt++] = dstRule;
|
||||
}
|
||||
trscount = cnt;
|
||||
}
|
||||
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
|
622
icu4c/source/i18n/tzrule.cpp
Normal file
622
icu4c/source/i18n/tzrule.cpp
Normal file
|
@ -0,0 +1,622 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/tzrule.h"
|
||||
#include "unicode/ucal.h"
|
||||
#include "gregoimp.h"
|
||||
#include "cmemory.h"
|
||||
#include "uarrsort.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
TimeZoneRule::TimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings)
|
||||
: UObject(), fName(name), fRawOffset(rawOffset), fDSTSavings(dstSavings) {
|
||||
}
|
||||
|
||||
TimeZoneRule::TimeZoneRule(const TimeZoneRule& source)
|
||||
: UObject(source), fName(source.fName), fRawOffset(source.fRawOffset), fDSTSavings(source.fDSTSavings) {
|
||||
}
|
||||
|
||||
TimeZoneRule::~TimeZoneRule() {
|
||||
}
|
||||
|
||||
TimeZoneRule&
|
||||
TimeZoneRule::operator=(const TimeZoneRule& right) {
|
||||
if (this != &right) {
|
||||
fName = right.fName;
|
||||
fRawOffset = right.fRawOffset;
|
||||
fDSTSavings = right.fDSTSavings;
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeZoneRule::operator==(const TimeZoneRule& that) const {
|
||||
return ((this == &that) ||
|
||||
(getDynamicClassID() == that.getDynamicClassID() &&
|
||||
fName == that.fName &&
|
||||
fRawOffset == that.fRawOffset &&
|
||||
fDSTSavings == that.fDSTSavings));
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeZoneRule::operator!=(const TimeZoneRule& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
UnicodeString&
|
||||
TimeZoneRule::getName(UnicodeString& name) const {
|
||||
name = fName;
|
||||
return name;
|
||||
}
|
||||
|
||||
int32_t
|
||||
TimeZoneRule::getRawOffset(void) const {
|
||||
return fRawOffset;
|
||||
}
|
||||
|
||||
int32_t
|
||||
TimeZoneRule::getDSTSavings(void) const {
|
||||
return fDSTSavings;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeZoneRule::isEquivalentTo(const TimeZoneRule& other) const {
|
||||
return ((this == &other) ||
|
||||
(getDynamicClassID() == other.getDynamicClassID() &&
|
||||
fRawOffset == other.fRawOffset &&
|
||||
fDSTSavings == other.fDSTSavings));
|
||||
}
|
||||
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(InitialTimeZoneRule)
|
||||
|
||||
InitialTimeZoneRule::InitialTimeZoneRule(const UnicodeString& name,
|
||||
int32_t rawOffset,
|
||||
int32_t dstSavings)
|
||||
: TimeZoneRule(name, rawOffset, dstSavings) {
|
||||
}
|
||||
|
||||
InitialTimeZoneRule::InitialTimeZoneRule(const InitialTimeZoneRule& source)
|
||||
: TimeZoneRule(source) {
|
||||
}
|
||||
|
||||
InitialTimeZoneRule::~InitialTimeZoneRule() {
|
||||
}
|
||||
|
||||
InitialTimeZoneRule*
|
||||
InitialTimeZoneRule::clone(void) const {
|
||||
return new InitialTimeZoneRule(*this);
|
||||
}
|
||||
|
||||
InitialTimeZoneRule&
|
||||
InitialTimeZoneRule::operator=(const InitialTimeZoneRule& right) {
|
||||
if (this != &right) {
|
||||
TimeZoneRule::operator=(right);
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::operator==(const InitialTimeZoneRule& that) const {
|
||||
return ((this == &that) ||
|
||||
(getDynamicClassID() == that.getDynamicClassID() &&
|
||||
TimeZoneRule::operator==(that)));
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::operator!=(const InitialTimeZoneRule& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::isEquivalentTo(const TimeZoneRule& other) const {
|
||||
if (this == &other) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != other.getDynamicClassID() ||
|
||||
TimeZoneRule::isEquivalentTo(other) == FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::getFirstStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::getFinalStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::getNextStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
InitialTimeZoneRule::getPreviousStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(AnnualTimeZoneRule)
|
||||
|
||||
const int32_t AnnualTimeZoneRule::MAX_YEAR = 0x7FFFFFFF; /* max signed int32 */
|
||||
|
||||
AnnualTimeZoneRule::AnnualTimeZoneRule(const UnicodeString& name,
|
||||
int32_t rawOffset,
|
||||
int32_t dstSavings,
|
||||
const DateTimeRule& dateTimeRule,
|
||||
int32_t startYear,
|
||||
int32_t endYear)
|
||||
: TimeZoneRule(name, rawOffset, dstSavings), fDateTimeRule(new DateTimeRule(dateTimeRule)),
|
||||
fStartYear(startYear), fEndYear(endYear) {
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule::AnnualTimeZoneRule(const UnicodeString& name,
|
||||
int32_t rawOffset,
|
||||
int32_t dstSavings,
|
||||
DateTimeRule* dateTimeRule,
|
||||
int32_t startYear,
|
||||
int32_t endYear)
|
||||
: TimeZoneRule(name, rawOffset, dstSavings), fDateTimeRule(dateTimeRule),
|
||||
fStartYear(startYear), fEndYear(endYear) {
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule::AnnualTimeZoneRule(const AnnualTimeZoneRule& source)
|
||||
: TimeZoneRule(source), fDateTimeRule(new DateTimeRule(*(source.fDateTimeRule))),
|
||||
fStartYear(source.fStartYear), fEndYear(source.fEndYear) {
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule::~AnnualTimeZoneRule() {
|
||||
delete fDateTimeRule;
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule*
|
||||
AnnualTimeZoneRule::clone(void) const {
|
||||
return new AnnualTimeZoneRule(*this);
|
||||
}
|
||||
|
||||
AnnualTimeZoneRule&
|
||||
AnnualTimeZoneRule::operator=(const AnnualTimeZoneRule& right) {
|
||||
if (this != &right) {
|
||||
TimeZoneRule::operator=(right);
|
||||
delete fDateTimeRule;
|
||||
fDateTimeRule = right.fDateTimeRule->clone();
|
||||
fStartYear = right.fStartYear;
|
||||
fEndYear = right.fEndYear;
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::operator==(const AnnualTimeZoneRule& that) const {
|
||||
return ((this == &that) ||
|
||||
(getDynamicClassID() == that.getDynamicClassID() &&
|
||||
TimeZoneRule::operator==(that) &&
|
||||
fDateTimeRule == that.fDateTimeRule &&
|
||||
fStartYear == that.fStartYear &&
|
||||
fEndYear == that.fEndYear));
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::operator!=(const AnnualTimeZoneRule& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
const DateTimeRule*
|
||||
AnnualTimeZoneRule::getRule() const {
|
||||
return fDateTimeRule;
|
||||
}
|
||||
|
||||
int32_t
|
||||
AnnualTimeZoneRule::getStartYear() const {
|
||||
return fStartYear;
|
||||
}
|
||||
|
||||
int32_t
|
||||
AnnualTimeZoneRule::getEndYear() const {
|
||||
return fEndYear;
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::getStartInYear(int32_t year,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate &result) const {
|
||||
if (year < fStartYear || year > fEndYear) {
|
||||
return FALSE;
|
||||
}
|
||||
double ruleDay;
|
||||
DateTimeRule::DateRuleType type = fDateTimeRule->getDateRuleType();
|
||||
if (type == DateTimeRule::DOM) {
|
||||
ruleDay = Grego::fieldsToDay(year, fDateTimeRule->getRuleMonth(), fDateTimeRule->getRuleDayOfMonth());
|
||||
} else {
|
||||
UBool after = TRUE;
|
||||
if (type == DateTimeRule::DOW) {
|
||||
// Normalize DOW rule into DOW_GEQ_DOM or DOW_LEQ_DOM
|
||||
int32_t weeks = fDateTimeRule->getRuleWeekInMonth();
|
||||
if (weeks > 0) {
|
||||
ruleDay = Grego::fieldsToDay(year, fDateTimeRule->getRuleMonth(), 1);
|
||||
ruleDay += 7 * (weeks - 1);
|
||||
} else {
|
||||
after = FALSE;
|
||||
ruleDay = Grego::fieldsToDay(year, fDateTimeRule->getRuleMonth(),
|
||||
Grego::monthLength(year, fDateTimeRule->getRuleMonth()));
|
||||
ruleDay += 7 * (weeks + 1);
|
||||
}
|
||||
} else {
|
||||
int32_t month = fDateTimeRule->getRuleMonth();
|
||||
int32_t dom = fDateTimeRule->getRuleDayOfMonth();
|
||||
if (type == DateTimeRule::DOW_LEQ_DOM) {
|
||||
after = FALSE;
|
||||
// Handle Feb <=29
|
||||
if (month == UCAL_FEBRUARY && dom == 29 && !Grego::isLeapYear(year)) {
|
||||
dom--;
|
||||
}
|
||||
}
|
||||
ruleDay = Grego::fieldsToDay(year, month, dom);
|
||||
}
|
||||
int32_t dow = Grego::dayOfWeek(ruleDay);
|
||||
int32_t delta = fDateTimeRule->getRuleDayOfWeek() - dow;
|
||||
if (after) {
|
||||
delta = delta < 0 ? delta + 7 : delta;
|
||||
} else {
|
||||
delta = delta > 0 ? delta - 7 : delta;
|
||||
}
|
||||
ruleDay += delta;
|
||||
}
|
||||
|
||||
result = ruleDay*U_MILLIS_PER_DAY + fDateTimeRule->getRuleMillisInDay();
|
||||
if (fDateTimeRule->getTimeRuleType() != DateTimeRule::UTC_TIME) {
|
||||
result -= prevRawOffset;
|
||||
}
|
||||
if (fDateTimeRule->getTimeRuleType() == DateTimeRule::WALL_TIME) {
|
||||
result -= prevDSTSavings;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::isEquivalentTo(const TimeZoneRule& other) const {
|
||||
if (this == &other) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != other.getDynamicClassID() ||
|
||||
TimeZoneRule::isEquivalentTo(other) == FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
AnnualTimeZoneRule* that = (AnnualTimeZoneRule*)&other;
|
||||
return (fDateTimeRule == that->fDateTimeRule &&
|
||||
fStartYear == that->fStartYear &&
|
||||
fEndYear == that->fEndYear);
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::getFirstStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
return getStartInYear(fStartYear, prevRawOffset, prevDSTSavings, result);
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::getFinalStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
if (fEndYear == MAX_YEAR) {
|
||||
return FALSE;
|
||||
}
|
||||
return getStartInYear(fEndYear, prevRawOffset, prevDSTSavings, result);
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::getNextStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
int32_t year, month, dom, dow, doy, mid;
|
||||
Grego::timeToFields(base, year, month, dom, dow, doy, mid);
|
||||
if (year < fStartYear) {
|
||||
return getFirstStart(prevRawOffset, prevDSTSavings, result);
|
||||
}
|
||||
UDate tmp;
|
||||
if (getStartInYear(year, prevRawOffset, prevDSTSavings, tmp)) {
|
||||
if (tmp < base || (!inclusive && (tmp == base))) {
|
||||
// Return the next one
|
||||
return getStartInYear(year + 1, prevRawOffset, prevDSTSavings, result);
|
||||
} else {
|
||||
result = tmp;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
AnnualTimeZoneRule::getPreviousStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
int32_t year, month, dom, dow, doy, mid;
|
||||
Grego::timeToFields(base, year, month, dom, dow, doy, mid);
|
||||
if (year > fEndYear) {
|
||||
return getFinalStart(prevRawOffset, prevDSTSavings, result);
|
||||
}
|
||||
UDate tmp;
|
||||
if (getStartInYear(year, prevRawOffset, prevDSTSavings, tmp)) {
|
||||
if (tmp > base || (!inclusive && (tmp == base))) {
|
||||
// Return the previous one
|
||||
return getStartInYear(year - 1, prevRawOffset, prevDSTSavings, result);
|
||||
} else {
|
||||
result = tmp;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(TimeArrayTimeZoneRule)
|
||||
|
||||
// UComparator function for sorting start times
|
||||
U_CFUNC int32_t
|
||||
compareDates(const void * /*context*/, const void *left, const void *right) {
|
||||
UDate l = *((UDate*)left);
|
||||
UDate r = *((UDate*)right);
|
||||
int32_t res = l < r ? -1 : (l == r ? 0 : 1);
|
||||
return res;
|
||||
}
|
||||
|
||||
TimeArrayTimeZoneRule::TimeArrayTimeZoneRule(const UnicodeString& name,
|
||||
int32_t rawOffset,
|
||||
int32_t dstSavings,
|
||||
const UDate* startTimes,
|
||||
int32_t numStartTimes,
|
||||
DateTimeRule::TimeRuleType timeRuleType)
|
||||
: TimeZoneRule(name, rawOffset, dstSavings), fTimeRuleType(timeRuleType),
|
||||
fStartTimes(NULL) {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initStartTimes(startTimes, numStartTimes, status);
|
||||
//TODO - status?
|
||||
}
|
||||
|
||||
|
||||
TimeArrayTimeZoneRule::TimeArrayTimeZoneRule(const TimeArrayTimeZoneRule& source)
|
||||
: TimeZoneRule(source), fTimeRuleType(source.fTimeRuleType), fStartTimes(NULL) {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initStartTimes(source.fStartTimes, source.fNumStartTimes, status);
|
||||
//TODO - status?
|
||||
}
|
||||
|
||||
|
||||
TimeArrayTimeZoneRule::~TimeArrayTimeZoneRule() {
|
||||
if (fStartTimes != NULL && fStartTimes != fLocalStartTimes) {
|
||||
uprv_free(fStartTimes);
|
||||
}
|
||||
}
|
||||
|
||||
TimeArrayTimeZoneRule*
|
||||
TimeArrayTimeZoneRule::clone(void) const {
|
||||
return new TimeArrayTimeZoneRule(*this);
|
||||
}
|
||||
|
||||
|
||||
TimeArrayTimeZoneRule&
|
||||
TimeArrayTimeZoneRule::operator=(const TimeArrayTimeZoneRule& right) {
|
||||
if (this != &right) {
|
||||
TimeZoneRule::operator=(right);
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
initStartTimes(right.fStartTimes, right.fNumStartTimes, status);
|
||||
//TODO - status?
|
||||
fTimeRuleType = right.fTimeRuleType;
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::operator==(const TimeArrayTimeZoneRule& that) const {
|
||||
if (this == &that) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != that.getDynamicClassID()
|
||||
|| TimeZoneRule::operator==(that) == FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
if (fTimeRuleType != that.fTimeRuleType ||
|
||||
fNumStartTimes != that.fNumStartTimes) {
|
||||
return FALSE;
|
||||
}
|
||||
// Compare start times
|
||||
UBool res = TRUE;
|
||||
for (int32_t i = 0; i < fNumStartTimes; i++) {
|
||||
if (fStartTimes[i] != that.fStartTimes[i]) {
|
||||
res = FALSE;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::operator!=(const TimeArrayTimeZoneRule& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
DateTimeRule::TimeRuleType
|
||||
TimeArrayTimeZoneRule::getTimeType(void) const {
|
||||
return fTimeRuleType;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::getStartTimeAt(int32_t index, UDate& result) const {
|
||||
if (index >= fNumStartTimes || index < 0) {
|
||||
return FALSE;
|
||||
}
|
||||
result = fStartTimes[index];
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
int32_t
|
||||
TimeArrayTimeZoneRule::countStartTimes(void) const {
|
||||
return fNumStartTimes;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::isEquivalentTo(const TimeZoneRule& other) const {
|
||||
if (this == &other) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != other.getDynamicClassID()
|
||||
|| TimeZoneRule::isEquivalentTo(other) == FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
TimeArrayTimeZoneRule* that = (TimeArrayTimeZoneRule*)&other;
|
||||
if (fTimeRuleType != that->fTimeRuleType ||
|
||||
fNumStartTimes != that->fNumStartTimes) {
|
||||
return FALSE;
|
||||
}
|
||||
// Compare start times
|
||||
UBool res = TRUE;
|
||||
for (int32_t i = 0; i < fNumStartTimes; i++) {
|
||||
if (fStartTimes[i] != that->fStartTimes[i]) {
|
||||
res = FALSE;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::getFirstStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
if (fNumStartTimes <= 0 || fStartTimes == NULL) {
|
||||
return FALSE;
|
||||
}
|
||||
result = getUTC(fStartTimes[0], prevRawOffset, prevDSTSavings);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::getFinalStart(int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UDate& result) const {
|
||||
if (fNumStartTimes <= 0 || fStartTimes == NULL) {
|
||||
return FALSE;
|
||||
}
|
||||
result = getUTC(fStartTimes[fNumStartTimes - 1], prevRawOffset, prevDSTSavings);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::getNextStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
int32_t i = fNumStartTimes - 1;
|
||||
for (; i >= 0; i--) {
|
||||
UDate time = getUTC(fStartTimes[i], prevRawOffset, prevDSTSavings);
|
||||
if (time < base || (!inclusive && time == base)) {
|
||||
break;
|
||||
}
|
||||
result = time;
|
||||
}
|
||||
if (i == fNumStartTimes - 1) {
|
||||
return FALSE;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::getPreviousStart(const UDate base,
|
||||
int32_t prevRawOffset,
|
||||
int32_t prevDSTSavings,
|
||||
UBool inclusive,
|
||||
UDate& result) const {
|
||||
int32_t i = fNumStartTimes - 1;
|
||||
for (; i >= 0; i--) {
|
||||
UDate time = getUTC(fStartTimes[i], prevRawOffset, prevDSTSavings);
|
||||
if (time < base || (inclusive && time == base)) {
|
||||
result = time;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
// ---- private methods ------
|
||||
|
||||
UBool
|
||||
TimeArrayTimeZoneRule::initStartTimes(const UDate source[], int32_t size, UErrorCode& status) {
|
||||
// Free old array
|
||||
if (fStartTimes != NULL && fStartTimes != fLocalStartTimes) {
|
||||
uprv_free(fStartTimes);
|
||||
}
|
||||
// Allocate new one if needed
|
||||
if (size > TIMEARRAY_STACK_BUFFER_SIZE) {
|
||||
fStartTimes = (UDate*)uprv_malloc(sizeof(UDate)*size);
|
||||
if (fStartTimes == NULL) {
|
||||
status = U_MEMORY_ALLOCATION_ERROR;
|
||||
fNumStartTimes = 0;
|
||||
return FALSE;
|
||||
}
|
||||
} else {
|
||||
fStartTimes = (UDate*)fLocalStartTimes;
|
||||
}
|
||||
uprv_memcpy(fStartTimes, source, sizeof(UDate)*size);
|
||||
fNumStartTimes = size;
|
||||
// Sort dates
|
||||
uprv_sortArray(fStartTimes, fNumStartTimes, (int32_t)sizeof(UDate), compareDates, NULL, TRUE, &status);
|
||||
if (U_FAILURE(status)) {
|
||||
if (fStartTimes != NULL && fStartTimes != fLocalStartTimes) {
|
||||
uprv_free(fStartTimes);
|
||||
}
|
||||
fNumStartTimes = 0;
|
||||
return FALSE;
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
UDate
|
||||
TimeArrayTimeZoneRule::getUTC(UDate time, int32_t raw, int32_t dst) const {
|
||||
if (fTimeRuleType != DateTimeRule::UTC_TIME) {
|
||||
time -= raw;
|
||||
}
|
||||
if (fTimeRuleType == DateTimeRule::WALL_TIME) {
|
||||
time -= dst;
|
||||
}
|
||||
return time;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
//eof
|
||||
|
142
icu4c/source/i18n/tztrans.cpp
Normal file
142
icu4c/source/i18n/tztrans.cpp
Normal file
|
@ -0,0 +1,142 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/tzrule.h"
|
||||
#include "unicode/tztrans.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(TimeZoneTransition)
|
||||
|
||||
TimeZoneTransition::TimeZoneTransition(const UDate& time, const TimeZoneRule& from, const TimeZoneRule& to)
|
||||
: UObject(), fTime(time), fFrom(from.clone()), fTo(to.clone()) {
|
||||
}
|
||||
|
||||
TimeZoneTransition::TimeZoneTransition(const UDate& time, TimeZoneRule* from, TimeZoneRule* to)
|
||||
: UObject(), fTime(time), fFrom(from), fTo(to) {
|
||||
}
|
||||
|
||||
TimeZoneTransition::TimeZoneTransition()
|
||||
: UObject(), fTime(0), fFrom(NULL), fTo(NULL) {
|
||||
}
|
||||
|
||||
TimeZoneTransition::TimeZoneTransition(const TimeZoneTransition& source)
|
||||
: UObject(), fTime(source.fTime), fFrom(source.fFrom->clone()),
|
||||
fTo(source.fTo->clone()) {
|
||||
}
|
||||
|
||||
TimeZoneTransition::~TimeZoneTransition() {
|
||||
if (fFrom != NULL) {
|
||||
delete fFrom;
|
||||
}
|
||||
if (fTo != NULL) {
|
||||
delete fTo;
|
||||
}
|
||||
}
|
||||
|
||||
TimeZoneTransition*
|
||||
TimeZoneTransition::clone(void) const {
|
||||
return new TimeZoneTransition(*this);
|
||||
}
|
||||
|
||||
TimeZoneTransition&
|
||||
TimeZoneTransition::operator=(const TimeZoneTransition& right) {
|
||||
if (this != &right) {
|
||||
fTime = right.fTime;
|
||||
setFrom(*right.fFrom);
|
||||
setTo(*right.fTo);
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeZoneTransition::operator==(const TimeZoneTransition& that) const {
|
||||
if (this == &that) {
|
||||
return TRUE;
|
||||
}
|
||||
if (getDynamicClassID() != that.getDynamicClassID()) {
|
||||
return FALSE;
|
||||
}
|
||||
if (fTime != that.fTime) {
|
||||
return FALSE;
|
||||
}
|
||||
if (fFrom == NULL && that.fFrom == NULL
|
||||
|| fFrom != NULL && that.fFrom != NULL && fFrom == that.fFrom) {
|
||||
if (fTo == NULL && that.fTo == NULL
|
||||
|| fTo != NULL && that.fTo != NULL && fTo == that.fTo) {
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
UBool
|
||||
TimeZoneTransition::operator!=(const TimeZoneTransition& that) const {
|
||||
return !operator==(that);
|
||||
}
|
||||
|
||||
void
|
||||
TimeZoneTransition::setTime(UDate time) {
|
||||
fTime = time;
|
||||
}
|
||||
|
||||
void
|
||||
TimeZoneTransition::setFrom(const TimeZoneRule& from) {
|
||||
if (fFrom != NULL) {
|
||||
delete fFrom;
|
||||
}
|
||||
fFrom = from.clone();
|
||||
}
|
||||
|
||||
void
|
||||
TimeZoneTransition::adoptFrom(TimeZoneRule* from) {
|
||||
if (fFrom != NULL) {
|
||||
delete fFrom;
|
||||
}
|
||||
fFrom = from;
|
||||
}
|
||||
|
||||
void
|
||||
TimeZoneTransition::setTo(const TimeZoneRule& to) {
|
||||
if (fTo != NULL) {
|
||||
delete fTo;
|
||||
}
|
||||
fTo = to.clone();
|
||||
}
|
||||
|
||||
void
|
||||
TimeZoneTransition::adoptTo(TimeZoneRule* to) {
|
||||
if (fTo != NULL) {
|
||||
delete fTo;
|
||||
}
|
||||
fTo = to;
|
||||
}
|
||||
|
||||
UDate
|
||||
TimeZoneTransition::getTime(void) const {
|
||||
return fTime;
|
||||
}
|
||||
|
||||
const TimeZoneRule*
|
||||
TimeZoneTransition::getTo(void) const {
|
||||
return fTo;
|
||||
}
|
||||
|
||||
const TimeZoneRule*
|
||||
TimeZoneTransition::getFrom(void) const {
|
||||
return fFrom;
|
||||
}
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif
|
||||
|
||||
//eof
|
181
icu4c/source/i18n/unicode/basictz.h
Normal file
181
icu4c/source/i18n/unicode/basictz.h
Normal file
|
@ -0,0 +1,181 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef BASICTZ_H
|
||||
#define BASICTZ_H
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: ICU TimeZone base class
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/timezone.h"
|
||||
#include "unicode/tzrule.h"
|
||||
#include "unicode/tztrans.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
// forward declarations
|
||||
class UVector;
|
||||
|
||||
/**
|
||||
* <code>BasicTimeZone</code> is an abstract class extending <code>TimeZone</code>.
|
||||
* This class provides some additional methods to access time zone transitions and rules.
|
||||
* All ICU <code>TimeZone</code> concrete subclasses extend this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API BasicTimeZone: public TimeZone {
|
||||
public:
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~BasicTimeZone();
|
||||
|
||||
/**
|
||||
* Gets the first time zone transition after the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the first transition after the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ = 0;
|
||||
|
||||
/**
|
||||
* Gets the most recent time zone transition before the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the most recent transition before the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ = 0;
|
||||
|
||||
/**
|
||||
* Checks if the time zone has equivalent transitions in the time range.
|
||||
* This method returns true when all of transition times, from/to standard
|
||||
* offsets and DST savings used by this time zone match the other in the
|
||||
* time range.
|
||||
* @param tz The <code>BasicTimeZone</code> object to be compared with.
|
||||
* @param start The start time of the evaluated time range (inclusive)
|
||||
* @param end The end time of the evaluated time range (inclusive)
|
||||
* @param ignoreDstAmount
|
||||
* When true, any transitions with only daylight saving amount
|
||||
* changes will be ignored, except either of them is zero.
|
||||
* For example, a transition from rawoffset 3:00/dstsavings 1:00
|
||||
* to rawoffset 2:00/dstsavings 2:00 is excluded from the comparison,
|
||||
* but a transtion from rawoffset 2:00/dstsavings 1:00 to
|
||||
* rawoffset 3:00/dstsavings 0:00 is included.
|
||||
* @param ec Output param to filled in with a success or an error.
|
||||
* @return true if the other time zone has the equivalent transitions in the
|
||||
* time range.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool hasEquivalentTransitions(/*const*/ BasicTimeZone& tz, UDate start, UDate end,
|
||||
UBool ignoreDstAmount, UErrorCode& ec) /*const*/;
|
||||
|
||||
/**
|
||||
* Returns the number of <code>TimeZoneRule</code>s which represents time transitions,
|
||||
* for this time zone, that is, all <code>TimeZoneRule</code>s for this time zone except
|
||||
* <code>InitialTimeZoneRule</code>. The return value range is 0 or any positive value.
|
||||
* @param status Receives error status code.
|
||||
* @return The number of <code>TimeZoneRule</code>s representing time transitions.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t countTransitionRules(UErrorCode& status) /*const*/ = 0;
|
||||
|
||||
/**
|
||||
* Gets the <code>InitialTimeZoneRule</code> and the set of <code>TimeZoneRule</code>
|
||||
* which represent time transitions for this time zone. On successful return,
|
||||
* the argument initial points to non-NULL <code>InitialTimeZoneRule</code> and
|
||||
* the array trsrules is filled with 0 or multiple <code>TimeZoneRule</code>
|
||||
* instances up to the size specified by trscount. The results are referencing the
|
||||
* rule instance held by this time zone instance. Therefore, after this time zone
|
||||
* is destructed, they are no longer available.
|
||||
* @param initial Receives the initial timezone rule
|
||||
* @param trsrules Receives the timezone transition rules
|
||||
* @param trscount On input, specify the size of the array 'transitions' receiving
|
||||
* the timezone transition rules. On output, actual number of
|
||||
* rules filled in the array will be set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[], int32_t& trscount, UErrorCode& status) /*const*/ = 0;
|
||||
|
||||
/**
|
||||
* Gets the set of time zone rules valid at the specified time. Some known external time zone
|
||||
* implementations are not capable to handle historic time zone rule changes. Also some
|
||||
* implementations can only handle certain type of rule definitions.
|
||||
* If this time zone does not use any daylight saving time within about 1 year from the specified
|
||||
* time, only the <code>InitialTimeZone</code> is returned. Otherwise, the rule for standard
|
||||
* time and daylight saving time transitions are returned in addition to the
|
||||
* <code>InitialTimeZoneRule</code>. The standard and daylight saving time transition rules are
|
||||
* represented by <code>AnnualTimeZoneRule</code> with <code>DateTimeRule::DOW</code> for its date
|
||||
* rule and <code>DateTimeRule::WALL_TIME</code> for its time rule. Because daylight saving time
|
||||
* rule is changing time to time in many time zones and also mapping a transition time rule to
|
||||
* different type is lossy transformation, the set of rules returned by this method may be valid
|
||||
* for short period of time.
|
||||
* The time zone rule objects returned by this method is owned by the caller, so the caller is
|
||||
* responsible for deleting them after use.
|
||||
* @param date The date used for extracting time zone rules.
|
||||
* @param initial Receives the <code>InitialTimeZone</code>, always not NULL.
|
||||
* @param std Receives the <code>AnnualTimeZoneRule</code> for standard time transitions.
|
||||
* When this time time zone does not observe daylight saving times around the
|
||||
* specified date, NULL is set.
|
||||
* @param dst Receives the <code>AnnualTimeZoneRule</code> for daylight saving time
|
||||
* transitions. When this time zone does not observer daylight saving times
|
||||
* around the specified date, NULL is set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getSimpleRulesNear(UDate date, InitialTimeZoneRule*& initial,
|
||||
AnnualTimeZoneRule*& std, AnnualTimeZoneRule*& dst, UErrorCode& status) /*const*/;
|
||||
|
||||
protected:
|
||||
/**
|
||||
* Default constructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
BasicTimeZone();
|
||||
|
||||
/*
|
||||
* Construct a timezone with a given ID.
|
||||
* @param id a system time zone ID
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
BasicTimeZone(const UnicodeString &id);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source the object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
BasicTimeZone(const BasicTimeZone& source);
|
||||
|
||||
/**
|
||||
* Gets the set of TimeZoneRule instances applicable to the specified time and after.
|
||||
* @param start The start date used for extracting time zone rules
|
||||
* @param initial Receives the InitialTimeZone, always not NULL
|
||||
* @param transitionRules Receives the transition rules, could be NULL
|
||||
* @param status Receives error status code
|
||||
*/
|
||||
void getTimeZoneRulesAfter(UDate start, InitialTimeZoneRule*& initial, UVector*& transitionRules,
|
||||
UErrorCode& status) /*const*/;
|
||||
};
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // BASICTZ_H
|
||||
|
||||
//eof
|
250
icu4c/source/i18n/unicode/dtrule.h
Normal file
250
icu4c/source/i18n/unicode/dtrule.h
Normal file
|
@ -0,0 +1,250 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef DTRULE_H
|
||||
#define DTRULE_H
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: Rule for specifying date and time in an year
|
||||
*/
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/uobject.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
/**
|
||||
* <code>DateTimeRule</code> is a class representing a time in a year by
|
||||
* a rule specified by month, day of month, day of week and
|
||||
* time in the day.
|
||||
*
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API DateTimeRule : public UObject {
|
||||
public:
|
||||
|
||||
/**
|
||||
* Date rule type constants.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
enum DateRuleType {
|
||||
DOM = 0, /**< The exact day of month,
|
||||
for example, March 11. */
|
||||
DOW, /**< The Nth occurence of the day of week,
|
||||
for example, 2nd Sunday in March. */
|
||||
DOW_GEQ_DOM, /**< The first occurence of the day of week on or after the day of monnth,
|
||||
for example, first Sunday on or after March 8. */
|
||||
DOW_LEQ_DOM /**< The last occurence of the day of week on or before the day of month,
|
||||
for example, first Sunday on or before March 14. */
|
||||
};
|
||||
|
||||
/**
|
||||
* Time rule type constants.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
enum TimeRuleType {
|
||||
WALL_TIME = 0, /**< The local wall clock time */
|
||||
STANDARD_TIME, /**< The local standard time */
|
||||
UTC_TIME /**< The UTC time */
|
||||
};
|
||||
|
||||
/**
|
||||
* Constructs a <code>DateTimeRule</code> by the day of month and
|
||||
* the time rule. The date rule type for an instance created by
|
||||
* this constructor is <code>DOM</code>.
|
||||
*
|
||||
* @param month The rule month, for example, <code>Calendar::JANUARY</code>
|
||||
* @param dayOfMonth The day of month, 1-based.
|
||||
* @param millisInDay The milliseconds in the rule date.
|
||||
* @param timeType The time type, <code>WALL_TIME</code> or <code>STANDARD_TIME</code>
|
||||
* or <code>UTC_TIME</code>.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule(int32_t month, int32_t dayOfMonth,
|
||||
int32_t millisInDay, TimeRuleType timeType);
|
||||
|
||||
/**
|
||||
* Constructs a <code>DateTimeRule</code> by the day of week and its oridinal
|
||||
* number and the time rule. The date rule type for an instance created
|
||||
* by this constructor is <code>DOW</code>.
|
||||
*
|
||||
* @param month The rule month, for example, <code>Calendar::JANUARY</code>.
|
||||
* @param weekInMonth The ordinal number of the day of week. Negative number
|
||||
* may be used for specifying a rule date counted from the
|
||||
* end of the rule month.
|
||||
* @param dayOfWeek The day of week, for example, <code>Calendar::SUNDAY</code>.
|
||||
* @param millisInDay The milliseconds in the rule date.
|
||||
* @param timeType The time type, <code>WALL_TIME</code> or <code>STANDARD_TIME</code>
|
||||
* or <code>UTC_TIME</code>.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule(int32_t month, int32_t weekInMonth, int32_t dayOfWeek,
|
||||
int32_t millisInDay, TimeRuleType timeType);
|
||||
|
||||
/**
|
||||
* Constructs a <code>DateTimeRule</code> by the first/last day of week
|
||||
* on or after/before the day of month and the time rule. The date rule
|
||||
* type for an instance created by this constructor is either
|
||||
* <code>DOM_GEQ_DOM</code> or <code>DOM_LEQ_DOM</code>.
|
||||
*
|
||||
* @param month The rule month, for example, <code>Calendar::JANUARY</code>
|
||||
* @param dayOfMonth The day of month, 1-based.
|
||||
* @param dayOfWeek The day of week, for example, <code>Calendar::SUNDAY</code>.
|
||||
* @param after true if the rule date is on or after the day of month.
|
||||
* @param millisInDay The milliseconds in the rule date.
|
||||
* @param timeType The time type, <code>WALL_TIME</code> or <code>STANDARD_TIME</code>
|
||||
* or <code>UTC_TIME</code>.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule(int32_t month, int32_t dayOfMonth, int32_t dayOfWeek, UBool after,
|
||||
int32_t millisInDay, TimeRuleType timeType);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The DateTimeRule object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule(const DateTimeRule& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
~DateTimeRule();
|
||||
|
||||
/**
|
||||
* Clone this DateTimeRule object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule* clone(void) const;
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateTimeRule& operator=(const DateTimeRule& right);
|
||||
|
||||
/**
|
||||
* Return true if the given DateTimeRule objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given DateTimeRule objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool operator==(const DateTimeRule& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given DateTimeRule objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given DateTimeRule objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool operator!=(const DateTimeRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the date rule type, such as <code>DOM</code>
|
||||
* @return The date rule type.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
DateRuleType getDateRuleType(void) const;
|
||||
|
||||
/**
|
||||
* Gets the time rule type
|
||||
* @return The time rule type, either <code>WALL_TIME</code> or <code>STANDARD_TIME</code>
|
||||
* or <code>UTC_TIME</code>.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeRuleType getTimeRuleType(void) const;
|
||||
|
||||
/**
|
||||
* Gets the rule month.
|
||||
* @return The rule month.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRuleMonth(void) const;
|
||||
|
||||
/**
|
||||
* Gets the rule day of month. When the date rule type
|
||||
* is <code>DOW</code>, the value is always 0.
|
||||
* @return The rule day of month
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRuleDayOfMonth(void) const;
|
||||
|
||||
/**
|
||||
* Gets the rule day of week. When the date rule type
|
||||
* is <code>DOM</code>, the value is always 0.
|
||||
* @return The rule day of week.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRuleDayOfWeek(void) const;
|
||||
|
||||
/**
|
||||
* Gets the ordinal number of the occurence of the day of week
|
||||
* in the month. When the date rule type is not <code>DOW</code>,
|
||||
* the value is always 0.
|
||||
* @return The rule day of week ordinal number in the month.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRuleWeekInMonth(void) const;
|
||||
|
||||
/**
|
||||
* Gets the rule time in the rule day.
|
||||
* @return The time in the rule day in milliseconds.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRuleMillisInDay(void) const;
|
||||
|
||||
private:
|
||||
DateRuleType fDateRuleType;
|
||||
TimeRuleType fTimeRuleType;
|
||||
int32_t fMonth;
|
||||
int32_t fDayOfMonth;
|
||||
int32_t fDayOfWeek;
|
||||
int32_t fWeekInMonth;
|
||||
int32_t fMillisInDay;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // DTRULE_H
|
||||
//eof
|
340
icu4c/source/i18n/unicode/rbtz.h
Normal file
340
icu4c/source/i18n/unicode/rbtz.h
Normal file
|
@ -0,0 +1,340 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef RBTZ_H
|
||||
#define RBTZ_H
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: Rule based customizable time zone
|
||||
*/
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/basictz.h"
|
||||
#include "unicode/unistr.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
// forward declaration
|
||||
class UVector;
|
||||
|
||||
class U_I18N_API RuleBasedTimeZone : public BasicTimeZone {
|
||||
public:
|
||||
/**
|
||||
* Constructs a <code>RuleBasedTimeZone</code> object with the ID and the
|
||||
* <code>InitialTimeZoneRule</code>. The input <code>InitialTimeZoneRule</code>
|
||||
* is adopted by this <code>RuleBasedTimeZone</code>, thus the caller must not
|
||||
* delete it.
|
||||
* @param id The time zone ID.
|
||||
* @param initialRule The initial time zone rule.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
RuleBasedTimeZone(const UnicodeString& id, InitialTimeZoneRule* initialRule);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The RuleBasedTimeZone object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
RuleBasedTimeZone(const RuleBasedTimeZone& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~RuleBasedTimeZone();
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
RuleBasedTimeZone& operator=(const RuleBasedTimeZone& right);
|
||||
|
||||
/**
|
||||
* Return true if the given <code>RuleBasedTimeZone</code> objects are
|
||||
* semantically equal. Objects of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given <code>RuleBasedTimeZone</code> objects are
|
||||
*semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const RuleBasedTimeZone& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given <code>RuleBasedTimeZone</code> objects are
|
||||
* semantically unequal. Objects of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given <code>RuleBasedTimeZone</code> objects are
|
||||
* semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const RuleBasedTimeZone& that) const;
|
||||
|
||||
/**
|
||||
* Adds the <code>TimeZoneRule</code> which represents time transitions.
|
||||
* The <code>TimeZoneRule</code> must have start times, that is, the result
|
||||
* of isTransitionRule() must be true. Otherwise, U_ILLEGAL_ARGUMENT_ERROR
|
||||
* is set to the error code.
|
||||
* The input <code>TimeZoneRule</code> is adopted by this
|
||||
* <code>RuleBasedTimeZone</code> on successful completion of this method,
|
||||
* thus, the caller must not delete it when no error is returned.
|
||||
* After all rules are added, the caller must call complete() method to
|
||||
* make this <code>RuleBasedTimeZone</code> ready to handle common time
|
||||
* zone functions.
|
||||
* @param rule The <code>TimeZoneRule</code>.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void addTransitionRule(TimeZoneRule* rule, UErrorCode& status);
|
||||
|
||||
/**
|
||||
* Makes the <code>TimeZoneRule</code> ready to handle actual timezone
|
||||
* calcuation APIs. This method collects time zone rules specified
|
||||
* by the caller via the constructor and addTransitionRule() and
|
||||
* builds internal structure for making the object ready to support
|
||||
* time zone APIs such as getOffset(), getNextTransition() and others.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void complete(UErrorCode& status);
|
||||
|
||||
/**
|
||||
* Clones TimeZone objects polymorphically. Clients are responsible for deleting
|
||||
* the TimeZone object cloned.
|
||||
*
|
||||
* @return A new copy of this TimeZone object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual TimeZone* clone(void) const;
|
||||
|
||||
/**
|
||||
* Returns the TimeZone's adjusted GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time in this time zone, taking daylight savings time into
|
||||
* account) as of a particular reference date. The reference date is used to determine
|
||||
* whether daylight savings time is in effect and needs to be figured into the offset
|
||||
* that is returned (in other words, what is the adjusted GMT offset in this time zone
|
||||
* at this particular date and time?). For the time zones produced by createTimeZone(),
|
||||
* the reference data is specified according to the Gregorian calendar, and the date
|
||||
* and time fields are local standard time.
|
||||
*
|
||||
* <p>Note: Don't call this method. Instead, call the getOffset(UDate...) overload,
|
||||
* which returns both the raw and the DST offset for a given time. This method
|
||||
* is retained only for backward compatibility.
|
||||
*
|
||||
* @param era The reference date's era
|
||||
* @param year The reference date's year
|
||||
* @param month The reference date's month (0-based; 0 is January)
|
||||
* @param day The reference date's day-in-month (1-based)
|
||||
* @param dayOfWeek The reference date's day-of-week (1-based; 1 is Sunday)
|
||||
* @param millis The reference date's milliseconds in day, local standard time
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @return The offset in milliseconds to add to GMT to get local time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Gets the time zone offset, for current date, modified in case of
|
||||
* daylight savings. This is the offset to add *to* UTC to get local time.
|
||||
*
|
||||
* <p>Note: Don't call this method. Instead, call the getOffset(UDate...) overload,
|
||||
* which returns both the raw and the DST offset for a given time. This method
|
||||
* is retained only for backward compatibility.
|
||||
*
|
||||
* @param era The reference date's era
|
||||
* @param year The reference date's year
|
||||
* @param month The reference date's month (0-based; 0 is January)
|
||||
* @param day The reference date's day-in-month (1-based)
|
||||
* @param dayOfWeek The reference date's day-of-week (1-based; 1 is Sunday)
|
||||
* @param millis The reference date's milliseconds in day, local standard time
|
||||
* @param monthLength The length of the given month in days.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @return The offset in milliseconds to add to GMT to get local time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis,
|
||||
int32_t monthLength, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Returns the time zone raw and GMT offset for the given moment
|
||||
* in time. Upon return, local-millis = GMT-millis + rawOffset +
|
||||
* dstOffset. All computations are performed in the proleptic
|
||||
* Gregorian calendar. The default implementation in the TimeZone
|
||||
* class delegates to the 8-argument getOffset().
|
||||
*
|
||||
* @param date moment in time for which to return offsets, in
|
||||
* units of milliseconds from January 1, 1970 0:00 GMT, either GMT
|
||||
* time or local wall time, depending on `local'.
|
||||
* @param local if true, `date' is local wall time; otherwise it
|
||||
* is in GMT time.
|
||||
* @param rawOffset output parameter to receive the raw offset, that
|
||||
* is, the offset not including DST adjustments
|
||||
* @param dstOffset output parameter to receive the DST offset,
|
||||
* that is, the offset to be added to `rawOffset' to obtain the
|
||||
* total offset between local and GMT time. If DST is not in
|
||||
* effect, this value is zero; otherwise it is a positive value,
|
||||
* typically one hour.
|
||||
* @param ec input-output error code
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getOffset(UDate date, UBool local, int32_t& rawOffset,
|
||||
int32_t& dstOffset, UErrorCode& ec) const;
|
||||
|
||||
/**
|
||||
* Sets the TimeZone's raw GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time, before taking daylight savings time into account).
|
||||
*
|
||||
* @param offsetMillis The new raw GMT offset for this time zone.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void setRawOffset(int32_t offsetMillis);
|
||||
|
||||
/**
|
||||
* Returns the TimeZone's raw GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time, before taking daylight savings time into account).
|
||||
*
|
||||
* @return The TimeZone's raw GMT offset.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getRawOffset(void) const;
|
||||
|
||||
/**
|
||||
* Queries if this time zone uses daylight savings time.
|
||||
* @return true if this time zone uses daylight savings time,
|
||||
* false, otherwise.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool useDaylightTime(void) const;
|
||||
|
||||
/**
|
||||
* Queries if the given date is in daylight savings time in
|
||||
* this time zone.
|
||||
* This method is wasteful since it creates a new GregorianCalendar and
|
||||
* deletes it each time it is called. This is a deprecated method
|
||||
* and provided only for Java compatibility.
|
||||
*
|
||||
* @param date the given UDate.
|
||||
* @param status Output param filled in with success/error code.
|
||||
* @return true if the given date is in daylight savings time,
|
||||
* false, otherwise.
|
||||
* @deprecated ICU 2.4. Use Calendar::inDaylightTime() instead.
|
||||
*/
|
||||
virtual UBool inDaylightTime(UDate date, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Returns true if this zone has the same rule and offset as another zone.
|
||||
* That is, if this zone differs only in ID, if at all.
|
||||
* @param other the <code>TimeZone</code> object to be compared with
|
||||
* @return true if the given zone is the same as this one,
|
||||
* with the possible exception of the ID
|
||||
* @stable ICU 3.8
|
||||
*/
|
||||
virtual UBool hasSameRules(const TimeZone& other) const;
|
||||
|
||||
/**
|
||||
* Gets the first time zone transition after the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the first transition after the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the most recent time zone transition before the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the most recent transition before the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Returns the number of <code>TimeZoneRule</code>s which represents time transitions,
|
||||
* for this time zone, that is, all <code>TimeZoneRule</code>s for this time zone except
|
||||
* <code>InitialTimeZoneRule</code>. The return value range is 0 or any positive value.
|
||||
* @param status Receives error status code.
|
||||
* @return The number of <code>TimeZoneRule</code>s representing time transitions.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t countTransitionRules(UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the <code>InitialTimeZoneRule</code> and the set of <code>TimeZoneRule</code>
|
||||
* which represent time transitions for this time zone. On successful return,
|
||||
* the argument initial points to non-NULL <code>InitialTimeZoneRule</code> and
|
||||
* the array trsrules is filled with 0 or multiple <code>TimeZoneRule</code>
|
||||
* instances up to the size specified by trscount. The results are referencing the
|
||||
* rule instance held by this time zone instance. Therefore, after this time zone
|
||||
* is destructed, they are no longer available.
|
||||
* @param initial Receives the initial timezone rule
|
||||
* @param trsrules Receives the timezone transition rules
|
||||
* @param trscount On input, specify the size of the array 'transitions' receiving
|
||||
* the timezone transition rules. On output, actual number of
|
||||
* rules filled in the array will be set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[], int32_t& trscount, UErrorCode& status) /*const*/;
|
||||
|
||||
private:
|
||||
void deleteRules(void);
|
||||
void deleteTransitions(void);
|
||||
UVector* copyRules(UVector* source);
|
||||
TimeZoneRule* findRuleInFinal(UDate date, UBool local) const;
|
||||
UBool findNext(UDate base, UBool inclusive, UDate& time, TimeZoneRule*& from, TimeZoneRule*& to) const;
|
||||
UBool findPrev(UDate base, UBool inclusive, UDate& time, TimeZoneRule*& from, TimeZoneRule*& to) const;
|
||||
|
||||
UBool fUpToDate;
|
||||
InitialTimeZoneRule *fInitialRule;
|
||||
UVector *fHistoricRules;
|
||||
UVector *fFinalRules;
|
||||
UVector *fHistoricTransitions;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // RBTZ_H
|
||||
|
||||
//eof
|
|
@ -32,10 +32,15 @@
|
|||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/timezone.h"
|
||||
#include "unicode/basictz.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
// forward declaration
|
||||
class InitialTimeZoneRule;
|
||||
class TimeZoneTransition;
|
||||
class AnnualTimeZoneRule;
|
||||
|
||||
/**
|
||||
* <code>SimpleTimeZone</code> is a concrete subclass of <code>TimeZone</code>
|
||||
* that represents a time zone for use with a Gregorian calendar. This
|
||||
|
@ -52,7 +57,7 @@ U_NAMESPACE_BEGIN
|
|||
* @see TimeZone
|
||||
* @author D. Goldsmith, Mark Davis, Chen-Lieh Huang, Alan Liu
|
||||
*/
|
||||
class U_I18N_API SimpleTimeZone: public TimeZone {
|
||||
class U_I18N_API SimpleTimeZone: public BasicTimeZone {
|
||||
public:
|
||||
|
||||
/**
|
||||
|
@ -688,6 +693,56 @@ public:
|
|||
*/
|
||||
virtual TimeZone* clone(void) const;
|
||||
|
||||
/**
|
||||
* Gets the first time zone transition after the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the first transition after the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the most recent time zone transition before the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the most recent transition before the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Returns the number of <code>TimeZoneRule</code>s which represents time transitions,
|
||||
* for this time zone, that is, all <code>TimeZoneRule</code>s for this time zone except
|
||||
* <code>InitialTimeZoneRule</code>. The return value range is 0 or any positive value.
|
||||
* @param status Receives error status code.
|
||||
* @return The number of <code>TimeZoneRule</code>s representing time transitions.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t countTransitionRules(UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the <code>InitialTimeZoneRule</code> and the set of <code>TimeZoneRule</code>
|
||||
* which represent time transitions for this time zone. On successful return,
|
||||
* the argument initial points to non-NULL <code>InitialTimeZoneRule</code> and
|
||||
* the array trsrules is filled with 0 or multiple <code>TimeZoneRule</code>
|
||||
* instances up to the size specified by trscount. The results are referencing the
|
||||
* rule instance held by this time zone instance. Therefore, after this time zone
|
||||
* is destructed, they are no longer available.
|
||||
* @param initial Receives the initial timezone rule
|
||||
* @param trsrules Receives the timezone transition rules
|
||||
* @param trscount On input, specify the size of the array 'transitions' receiving
|
||||
* the timezone transition rules. On output, actual number of
|
||||
* rules filled in the array will be set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[], int32_t& trscount, UErrorCode& status) /*const*/;
|
||||
|
||||
|
||||
public:
|
||||
|
||||
/**
|
||||
|
@ -805,6 +860,16 @@ private:
|
|||
* Typically one hour; sometimes 30 minutes.
|
||||
*/
|
||||
int32_t dstSavings;
|
||||
|
||||
/* Private for BasicTimeZone implementation */
|
||||
void initTransitionRules(UErrorCode& status);
|
||||
void clearTransitionRules(void);
|
||||
void deleteTransitionRules(void);
|
||||
UBool transitionRulesInitialized;
|
||||
InitialTimeZoneRule* initialRule;
|
||||
TimeZoneTransition* firstTransition;
|
||||
AnnualTimeZoneRule* stdRule;
|
||||
AnnualTimeZoneRule* dstRule;
|
||||
};
|
||||
|
||||
inline void SimpleTimeZone::setStartRule(int32_t month, int32_t dayOfWeekInMonth,
|
||||
|
|
826
icu4c/source/i18n/unicode/tzrule.h
Normal file
826
icu4c/source/i18n/unicode/tzrule.h
Normal file
|
@ -0,0 +1,826 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef TZRULE_H
|
||||
#define TZRULE_H
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: Time zone rule classes
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/uobject.h"
|
||||
#include "unicode/unistr.h"
|
||||
#include "unicode/dtrule.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
/**
|
||||
* <code>TimeZoneRule</code> is a class representing a rule for time zone.
|
||||
* <code>TimeZoneRule</code> has a set of time zone attributes, such as zone name,
|
||||
* raw offset (UTC offset for standard time) and daylight saving time offset.
|
||||
*
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API TimeZoneRule : public UObject {
|
||||
public:
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~TimeZoneRule();
|
||||
|
||||
/**
|
||||
* Clone this TimeZoneRule object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual TimeZoneRule* clone(void) const = 0;
|
||||
|
||||
/**
|
||||
* Return true if the given TimeZoneRule objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeZoneRule objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const TimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given TimeZoneRule objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeZoneRule objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const TimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Fills in "name" with the name of this time zone.
|
||||
* @param name Receives the name of this time zone.
|
||||
* @return A reference to "name"
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UnicodeString& getName(UnicodeString& name) const;
|
||||
|
||||
/**
|
||||
* Gets the standard time offset.
|
||||
* @return The standard time offset from UTC in milliseconds.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getRawOffset(void) const;
|
||||
|
||||
/**
|
||||
* Gets the amount of daylight saving delta time from the standard time.
|
||||
* @return The amount of daylight saving offset used by this rule
|
||||
* in milliseconds.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getDSTSavings(void) const;
|
||||
|
||||
/**
|
||||
* Returns if this rule represents the same rule and offsets as another.
|
||||
* When two <code>TimeZoneRule</code> objects differ only its names, this method
|
||||
* returns true.
|
||||
* @param other The <code>TimeZoneRule</code> object to be compared with.
|
||||
* @return true if the other <code>TimeZoneRule</code> is the same as this one.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool isEquivalentTo(const TimeZoneRule& other) const;
|
||||
|
||||
/**
|
||||
* Gets the very first time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the very first time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFirstStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const = 0;
|
||||
|
||||
/**
|
||||
* Gets the final time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the final time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFinalStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const = 0;
|
||||
|
||||
/**
|
||||
* Gets the first time when this rule takes effect after the specified time.
|
||||
* @param base The first start time after this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The first time when this rule takes effect after
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const = 0;
|
||||
|
||||
/**
|
||||
* Gets the most recent time when this rule takes effect before the specified time.
|
||||
* @param base The most recent time before this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The most recent time when this rule takes effect before
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const = 0;
|
||||
|
||||
protected:
|
||||
|
||||
/**
|
||||
* Constructs a <code>TimeZoneRule</code> with the name, the GMT offset of its
|
||||
* standard time and the amount of daylight saving offset adjustment.
|
||||
* @param name The time zone name.
|
||||
* @param rawOffset The UTC offset of its standard time in milliseconds.
|
||||
* @param dstSavings The amount of daylight saving offset adjustment in milliseconds.
|
||||
* If this ia a rule for standard time, the value of this argument is 0.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The TimeZoneRule object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneRule(const TimeZoneRule& source);
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual TimeZoneRule& operator=(const TimeZoneRule& right);
|
||||
|
||||
private:
|
||||
UnicodeString fName; // time name
|
||||
int32_t fRawOffset; // UTC offset of the standard time in milliseconds
|
||||
int32_t fDSTSavings; // DST saving amount in milliseconds
|
||||
};
|
||||
|
||||
/**
|
||||
* <code>InitialTimeZoneRule</code> represents a time zone rule
|
||||
* representing a time zone effective from the beginning and
|
||||
* has no actual start times.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API InitialTimeZoneRule : public TimeZoneRule {
|
||||
public:
|
||||
/**
|
||||
* Constructs an <code>InitialTimeZoneRule</code> with the name, the GMT offset of its
|
||||
* standard time and the amount of daylight saving offset adjustment.
|
||||
* @param name The time zone name.
|
||||
* @param rawOffset The UTC offset of its standard time in milliseconds.
|
||||
* @param dstSavings The amount of daylight saving offset adjustment in milliseconds.
|
||||
* If this ia a rule for standard time, the value of this argument is 0.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
InitialTimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The InitialTimeZoneRule object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
InitialTimeZoneRule(const InitialTimeZoneRule& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~InitialTimeZoneRule();
|
||||
|
||||
/**
|
||||
* Clone this InitialTimeZoneRule object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual InitialTimeZoneRule* clone(void) const;
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
InitialTimeZoneRule& operator=(const InitialTimeZoneRule& right);
|
||||
|
||||
/**
|
||||
* Return true if the given InitialTimeZoneRule objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given InitialTimeZoneRule objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const InitialTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given InitialTimeZoneRule objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given InitialTimeZoneRule objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const InitialTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the time when this rule takes effect in the given year.
|
||||
* @param year The Gregorian year, with 0 == 1 BCE, -1 == 2 BCE, etc.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the start time in the year.
|
||||
* @return true if this rule takes effect in the year and the result is set to
|
||||
* "result".
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool getStartInYear(int32_t year, int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Returns if this rule represents the same rule and offsets as another.
|
||||
* When two <code>TimeZoneRule</code> objects differ only its names, this method
|
||||
* returns true.
|
||||
* @param that The <code>TimeZoneRule</code> object to be compared with.
|
||||
* @return true if the other <code>TimeZoneRule</code> is equivalent to this one.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool isEquivalentTo(const TimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the very first time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the very first time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFirstStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the final time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the final time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFinalStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the first time when this rule takes effect after the specified time.
|
||||
* @param base The first start time after this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The first time when this rule takes effect after
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the most recent time when this rule takes effect before the specified time.
|
||||
* @param base The most recent time before this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The most recent time when this rule takes effect before
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
/**
|
||||
* <code>AnnualTimeZoneRule</code> is a class used for representing a time zone
|
||||
* rule which takes effect annually. The calenday system used for the rule is
|
||||
* is based on Gregorian calendar
|
||||
*
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API AnnualTimeZoneRule : public TimeZoneRule {
|
||||
public:
|
||||
/**
|
||||
* The constant representing the maximum year used for designating
|
||||
* a rule is permanent.
|
||||
*/
|
||||
static const int32_t MAX_YEAR;
|
||||
|
||||
/**
|
||||
* Constructs a <code>AnnualTimeZoneRule</code> with the name, the GMT offset of its
|
||||
* standard time, the amount of daylight saving offset adjustment, the annual start
|
||||
* time rule and the start/until years. The input DateTimeRule is copied by this
|
||||
* constructor, so the caller remains responsible for deleting the object.
|
||||
* @param name The time zone name.
|
||||
* @param rawOffset The GMT offset of its standard time in milliseconds.
|
||||
* @param dstSavings The amount of daylight saving offset adjustment in
|
||||
* milliseconds. If this ia a rule for standard time,
|
||||
* the value of this argument is 0.
|
||||
* @param dateTimeRule The start date/time rule repeated annually.
|
||||
* @param startYear The first year when this rule takes effect.
|
||||
* @param endYear The last year when this rule takes effect. If this
|
||||
* rule is effective forever in future, specify MAX_YEAR.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
AnnualTimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings,
|
||||
const DateTimeRule& dateTimeRule, int32_t startYear, int32_t endYear);
|
||||
|
||||
/**
|
||||
* Constructs a <code>AnnualTimeZoneRule</code> with the name, the GMT offset of its
|
||||
* standard time, the amount of daylight saving offset adjustment, the annual start
|
||||
* time rule and the start/until years. The input DateTimeRule object is adopted
|
||||
* by this object, therefore, the caller must not delete the object.
|
||||
* @param name The time zone name.
|
||||
* @param rawOffset The GMT offset of its standard time in milliseconds.
|
||||
* @param dstSavings The amount of daylight saving offset adjustment in
|
||||
* milliseconds. If this ia a rule for standard time,
|
||||
* the value of this argument is 0.
|
||||
* @param dateTimeRule The start date/time rule repeated annually.
|
||||
* @param startYear The first year when this rule takes effect.
|
||||
* @param endYear The last year when this rule takes effect. If this
|
||||
* rule is effective forever in future, specify MAX_YEAR.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
AnnualTimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings,
|
||||
DateTimeRule* dateTimeRule, int32_t startYear, int32_t endYear);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The AnnualTimeZoneRule object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
AnnualTimeZoneRule(const AnnualTimeZoneRule& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~AnnualTimeZoneRule();
|
||||
|
||||
/**
|
||||
* Clone this AnnualTimeZoneRule object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual AnnualTimeZoneRule* clone(void) const;
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
AnnualTimeZoneRule& operator=(const AnnualTimeZoneRule& right);
|
||||
|
||||
/**
|
||||
* Return true if the given AnnualTimeZoneRule objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given AnnualTimeZoneRule objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const AnnualTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given AnnualTimeZoneRule objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given AnnualTimeZoneRule objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const AnnualTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the start date/time rule used by this rule.
|
||||
* @return The <code>AnnualDateTimeRule</code> which represents the start date/time
|
||||
* rule used by this time zone rule.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
const DateTimeRule* getRule(void) const;
|
||||
|
||||
/**
|
||||
* Gets the first year when this rule takes effect.
|
||||
* @return The start year of this rule. The year is in Gregorian calendar
|
||||
* with 0 == 1 BCE, -1 == 2 BCE, etc.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getStartYear(void) const;
|
||||
|
||||
/**
|
||||
* Gets the end year when this rule takes effect.
|
||||
* @return The end year of this rule (inclusive). The year is in Gregorian calendar
|
||||
* with 0 == 1 BCE, -1 == 2 BCE, etc.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
int32_t getEndYear(void) const;
|
||||
|
||||
/**
|
||||
* Gets the time when this rule takes effect in the given year.
|
||||
* @param year The Gregorian year, with 0 == 1 BCE, -1 == 2 BCE, etc.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the start time in the year.
|
||||
* @return true if this rule takes effect in the year and the result is set to
|
||||
* "result".
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool getStartInYear(int32_t year, int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Returns if this rule represents the same rule and offsets as another.
|
||||
* When two <code>TimeZoneRule</code> objects differ only its names, this method
|
||||
* returns true.
|
||||
* @param that The <code>TimeZoneRule</code> object to be compared with.
|
||||
* @return true if the other <code>TimeZoneRule</code> is equivalent to this one.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool isEquivalentTo(const TimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the very first time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the very first time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFirstStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the final time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the final time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFinalStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the first time when this rule takes effect after the specified time.
|
||||
* @param base The first start time after this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The first time when this rule takes effect after
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the most recent time when this rule takes effect before the specified time.
|
||||
* @param base The most recent time before this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The most recent time when this rule takes effect before
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
|
||||
private:
|
||||
DateTimeRule* fDateTimeRule;
|
||||
int32_t fStartYear;
|
||||
int32_t fEndYear;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
/**
|
||||
* <code>TimeArrayTimeZoneRule</code> represents a time zone rule whose start times are
|
||||
* defined by an array of milliseconds since the standard base time.
|
||||
*
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API TimeArrayTimeZoneRule : public TimeZoneRule {
|
||||
public:
|
||||
/**
|
||||
* Constructs a <code>TimeArrayTimeZoneRule</code> with the name, the GMT offset of its
|
||||
* standard time, the amount of daylight saving offset adjustment and
|
||||
* the array of times when this rule takes effect.
|
||||
* @param name The time zone name.
|
||||
* @param rawOffset The UTC offset of its standard time in milliseconds.
|
||||
* @param dstSavings The amount of daylight saving offset adjustment in
|
||||
* milliseconds. If this ia a rule for standard time,
|
||||
* the value of this argument is 0.
|
||||
* @param startTimes The array start times in milliseconds since the base time
|
||||
* (January 1, 1970, 00:00:00).
|
||||
* @param numStartTimes The number of elements in the parameter "startTimes"
|
||||
* @param timeRuleType The time type of the start times, which is one of
|
||||
* <code>DataTimeRule::WALL_TIME</code>, <code>STANDARD_TIME</code>
|
||||
* and <code>UNIVERSAL_TIME</code>.
|
||||
* @param status An UErrorCode to receive the stauts.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeArrayTimeZoneRule(const UnicodeString& name, int32_t rawOffset, int32_t dstSavings,
|
||||
const UDate* startTimes, int32_t numStartTimes, DateTimeRule::TimeRuleType timeRuleType);
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The TimeArrayTimeZoneRule object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeArrayTimeZoneRule(const TimeArrayTimeZoneRule& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~TimeArrayTimeZoneRule();
|
||||
|
||||
/**
|
||||
* Clone this TimeArrayTimeZoneRule object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual TimeArrayTimeZoneRule* clone(void) const;
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeArrayTimeZoneRule& operator=(const TimeArrayTimeZoneRule& right);
|
||||
|
||||
/**
|
||||
* Return true if the given TimeArrayTimeZoneRule objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeArrayTimeZoneRule objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const TimeArrayTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given TimeArrayTimeZoneRule objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeArrayTimeZoneRule objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const TimeArrayTimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the time type of the start times used by this rule. The return value
|
||||
* is either <code>DateTimeRule::WALL_TIME</code> or <code>STANDARD_TIME</code>
|
||||
* or <code>UNIVERSAL_TIME</code>.
|
||||
*
|
||||
* @return The time type used of the start times used by this rule.
|
||||
*/
|
||||
DateTimeRule::TimeRuleType getTimeType(void) const;
|
||||
|
||||
/**
|
||||
* Gets a start time at the index stored in this rule.
|
||||
* @param index The index of start times
|
||||
* @param result Receives the start time at the index
|
||||
* @return true if the index is within the valid range and
|
||||
* and the result is set. When false, the output
|
||||
* parameger "result" is unchanged.
|
||||
*/
|
||||
UBool getStartTimeAt(int32_t index, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Returns the number of start times stored in this rule
|
||||
* @return The number of start times.
|
||||
*/
|
||||
int32_t countStartTimes(void) const;
|
||||
|
||||
/**
|
||||
* Returns if this rule represents the same rule and offsets as another.
|
||||
* When two <code>TimeZoneRule</code> objects differ only its names, this method
|
||||
* returns true.
|
||||
* @param that The <code>TimeZoneRule</code> object to be compared with.
|
||||
* @return true if the other <code>TimeZoneRule</code> is equivalent to this one.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool isEquivalentTo(const TimeZoneRule& that) const;
|
||||
|
||||
/**
|
||||
* Gets the very first time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the very first time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFirstStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the final time when this rule takes effect.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param result Receives the final time when this rule takes effect.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getFinalStart(int32_t prevRawOffset, int32_t prevDSTSavings, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the first time when this rule takes effect after the specified time.
|
||||
* @param base The first start time after this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The first time when this rule takes effect after
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
/**
|
||||
* Gets the most recent time when this rule takes effect before the specified time.
|
||||
* @param base The most recent time before this base time will be returned.
|
||||
* @param prevRawOffset The standard time offset from UTC before this rule
|
||||
* takes effect in milliseconds.
|
||||
* @param prevDSTSavings The amount of daylight saving offset from the
|
||||
* standard time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives The most recent time when this rule takes effect before
|
||||
* the specified base time.
|
||||
* @return true if the start time is available. When false is returned, output parameter
|
||||
* "result" is unchanged.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousStart(UDate base, int32_t prevRawOffset, int32_t prevDSTSavings,
|
||||
UBool inclusive, UDate& result) const;
|
||||
|
||||
|
||||
private:
|
||||
enum { TIMEARRAY_STACK_BUFFER_SIZE = 32 };
|
||||
UBool initStartTimes(const UDate source[], int32_t size, UErrorCode& ec);
|
||||
UDate getUTC(UDate time, int32_t raw, int32_t dst) const;
|
||||
|
||||
DateTimeRule::TimeRuleType fTimeRuleType;
|
||||
int32_t fNumStartTimes;
|
||||
UDate* fStartTimes;
|
||||
UDate fLocalStartTimes[TIMEARRAY_STACK_BUFFER_SIZE];
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // TZRULE_H
|
||||
|
||||
//eof
|
190
icu4c/source/i18n/unicode/tztrans.h
Normal file
190
icu4c/source/i18n/unicode/tztrans.h
Normal file
|
@ -0,0 +1,190 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef TZTRANS_H
|
||||
#define TZTRANS_H
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: Time zone transition
|
||||
*/
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/uobject.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
// Forward declaration
|
||||
class TimeZoneRule;
|
||||
|
||||
/**
|
||||
* <code>TimeZoneTransition</code> is a class representing a time zone transition.
|
||||
* An instance has a time of transition and rules for both before and after the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API TimeZoneTransition : public UObject {
|
||||
public:
|
||||
/**
|
||||
*
|
||||
*/
|
||||
TimeZoneTransition(const UDate& time, const TimeZoneRule& from, const TimeZoneRule& to);
|
||||
TimeZoneTransition(const UDate& time, TimeZoneRule* from, TimeZoneRule* to);
|
||||
|
||||
/**
|
||||
* Constructs an empty <code>TimeZoneTransition</code>
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneTransition();
|
||||
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The TimeZoneTransition object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneTransition(const TimeZoneTransition& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
~TimeZoneTransition();
|
||||
|
||||
/**
|
||||
* Clone this TimeZoneTransition object polymorphically. The caller owns the result and
|
||||
* should delete it when done.
|
||||
* @return A copy of the object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneTransition* clone(void) const;
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
TimeZoneTransition& operator=(const TimeZoneTransition& right);
|
||||
|
||||
/**
|
||||
* Return true if the given TimeZoneTransition objects are semantically equal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeZoneTransition objects are semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool operator==(const TimeZoneTransition& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given TimeZoneTransition objects are semantically unequal. Objects
|
||||
* of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given TimeZoneTransition objects are semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool operator!=(const TimeZoneTransition& that) const;
|
||||
|
||||
/**
|
||||
* Returns the time of transition in milliseconds.
|
||||
* @return The time of the transition in milliseconds since the 1970 Jan 1 epoch time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UDate getTime(void) const;
|
||||
|
||||
/**
|
||||
* Sets the time of transition in milliseconds.
|
||||
* @param time The time of the transition in milliseconds since the 1970 Jan 1 epoch time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void setTime(UDate time);
|
||||
|
||||
/**
|
||||
* Returns the rule used before the transition.
|
||||
* @return The time zone rule used after the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
const TimeZoneRule* getFrom(void) const;
|
||||
|
||||
/**
|
||||
* Sets the rule used before the transition. The caller remains
|
||||
* responsible for deleting the <code>TimeZoneRule</code> object.
|
||||
* @param from The time zone rule used before the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void setFrom(const TimeZoneRule& from);
|
||||
|
||||
/**
|
||||
* Adopts the rule used before the transition. The caller must
|
||||
* not delete the <code>TimeZoneRule</code> object passed in.
|
||||
* @param from The time zone rule used before the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void adoptFrom(TimeZoneRule* from);
|
||||
|
||||
/**
|
||||
* Sets the rule used after the transition. The caller remains
|
||||
* responsible for deleting the <code>TimeZoneRule</code> object.
|
||||
* @param to The time zone rule used after the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void setTo(const TimeZoneRule& to);
|
||||
|
||||
/**
|
||||
* Adopts the rule used after the transition. The caller must
|
||||
* not delete the <code>TimeZoneRule</code> object passed in.
|
||||
* @param to The time zone rule used after the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void adoptTo(TimeZoneRule* to);
|
||||
|
||||
/**
|
||||
* Returns the rule used after the transition.
|
||||
* @return The time zone rule used after the transition.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
const TimeZoneRule* getTo(void) const;
|
||||
|
||||
private:
|
||||
UDate fTime;
|
||||
TimeZoneRule* fFrom;
|
||||
TimeZoneRule* fTo;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // TZTRANS_H
|
||||
|
||||
//eof
|
442
icu4c/source/i18n/unicode/vtzone.h
Normal file
442
icu4c/source/i18n/unicode/vtzone.h
Normal file
|
@ -0,0 +1,442 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
#ifndef VTZONE_H
|
||||
#define VTZONE_H
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
|
||||
/**
|
||||
* \file
|
||||
* \brief C++ API: RFC2445 VTIMEZONE support
|
||||
*/
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
#include "unicode/basictz.h"
|
||||
|
||||
U_NAMESPACE_BEGIN
|
||||
|
||||
class VTZWriter;
|
||||
class VTZReader;
|
||||
class UVector;
|
||||
|
||||
/**
|
||||
* <code>VTimeZone</code> is a class implementing RFC2445 VTIMEZONE. You can create a
|
||||
* <code>VTimeZone</code> instance from a time zone ID supported by <code>TimeZone</code>.
|
||||
* With the <code>VTimeZone</code> instance created from the ID, you can write out the rule
|
||||
* in RFC2445 VTIMEZONE format. Also, you can create a <code>VTimeZone</code> instance
|
||||
* from RFC2445 VTIMEZONE data stream, which allows you to calculate time
|
||||
* zone offset by the rules defined by the data.<br><br>
|
||||
* Note: The consumer of this class reading or writing VTIMEZONE data is responsible to
|
||||
* decode or encode Non-ASCII text. Methods reading/writing VTIMEZONE data in this class
|
||||
* do nothing with MIME encoding.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
class U_I18N_API VTimeZone : public BasicTimeZone {
|
||||
public:
|
||||
/**
|
||||
* Copy constructor.
|
||||
* @param source The <code>VTimeZone</code> object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
VTimeZone(const VTimeZone& source);
|
||||
|
||||
/**
|
||||
* Destructor.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual ~VTimeZone();
|
||||
|
||||
/**
|
||||
* Assignment operator.
|
||||
* @param right The object to be copied.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
VTimeZone& operator=(const VTimeZone& right);
|
||||
|
||||
/**
|
||||
* Return true if the given <code>VTimeZone</code> objects are
|
||||
* semantically equal. Objects of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given <code>VTimeZone</code> objects are
|
||||
*semantically equal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator==(const VTimeZone& that) const;
|
||||
|
||||
/**
|
||||
* Return true if the given <code>VTimeZone</code> objects are
|
||||
* semantically unequal. Objects of different subclasses are considered unequal.
|
||||
* @param that The object to be compared with.
|
||||
* @return true if the given <code>VTimeZone</code> objects are
|
||||
* semantically unequal.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool operator!=(const VTimeZone& that) const;
|
||||
|
||||
/**
|
||||
* Create a <code>VTimeZone</code> instance by the time zone ID.
|
||||
* @param ID The time zone ID, such as America/New_York
|
||||
* @return A <code>VTimeZone</code> object initialized by the time zone ID,
|
||||
* or NULL when the ID is unknown.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static VTimeZone* createVTimeZoneByID(const UnicodeString& ID);
|
||||
|
||||
/**
|
||||
* Create a <code>VTimeZone</code> instance by RFC2445 VTIMEZONE data
|
||||
*
|
||||
* @param vtzdata The string including VTIMEZONE data block
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @return A <code>VTimeZone</code> initialized by the VTIMEZONE data or
|
||||
* NULL if failed to load the rule from the VTIMEZONE data.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static VTimeZone* createVTimeZone(const UnicodeString& vtzdata, UErrorCode& status);
|
||||
|
||||
/**
|
||||
* Gets the RFC2445 TZURL property value. When a <code>VTimeZone</code> instance was
|
||||
* created from VTIMEZONE data, the initial value is set by the TZURL property value
|
||||
* in the data. Otherwise, the initial value is not set.
|
||||
* @param url Receives the RFC2445 TZURL property value.
|
||||
* @return TRUE if TZURL attribute is available and value is set.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool getTZURL(UnicodeString& url) const;
|
||||
|
||||
/**
|
||||
* Sets the RFC2445 TZURL property value.
|
||||
* @param url The TZURL property value.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void setTZURL(const UnicodeString& url);
|
||||
|
||||
/**
|
||||
* Gets the RFC2445 LAST-MODIFIED property value. When a <code>VTimeZone</code> instance
|
||||
* was created from VTIMEZONE data, the initial value is set by the LAST-MODIFIED property
|
||||
* value in the data. Otherwise, the initial value is not set.
|
||||
* @param lastModified Receives the last modified date.
|
||||
* @return TRUE if lastModified attribute is available and value is set.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
UBool getLastModified(UDate& lastModified) const;
|
||||
|
||||
/**
|
||||
* Sets the RFC2445 LAST-MODIFIED property value.
|
||||
* @param lastModified The LAST-MODIFIED date.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void setLastModified(UDate lastModified);
|
||||
|
||||
/**
|
||||
* Writes RFC2445 VTIMEZONE data for this time zone
|
||||
* @param result Output param to filled in with the VTIMEZONE data.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void write(UnicodeString& result, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Writes RFC2445 VTIMEZONE data for this time zone applicalbe
|
||||
* for dates after the specified start time.
|
||||
* @param start The start date.
|
||||
* @param result Output param to filled in with the VTIMEZONE data.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void write(UDate start, UnicodeString& result, UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Writes RFC2445 VTIMEZONE data applicalbe for the specified date.
|
||||
* Some common iCalendar implementations can only handle a single time
|
||||
* zone property or a pair of standard and daylight time properties using
|
||||
* BYDAY rule with day of week (such as BYDAY=1SUN). This method produce
|
||||
* the VTIMEZONE data which can be handled these implementations. The rules
|
||||
* produced by this method can be used only for calculating time zone offset
|
||||
* around the specified date.
|
||||
* @param time The date used for rule extraction.
|
||||
* @param result Output param to filled in with the VTIMEZONE data.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
void writeSimple(UDate time, UnicodeString& result, UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Clones TimeZone objects polymorphically. Clients are responsible for deleting
|
||||
* the TimeZone object cloned.
|
||||
* @return A new copy of this TimeZone object.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual TimeZone* clone(void) const;
|
||||
|
||||
/**
|
||||
* Returns the TimeZone's adjusted GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time in this time zone, taking daylight savings time into
|
||||
* account) as of a particular reference date. The reference date is used to determine
|
||||
* whether daylight savings time is in effect and needs to be figured into the offset
|
||||
* that is returned (in other words, what is the adjusted GMT offset in this time zone
|
||||
* at this particular date and time?). For the time zones produced by createTimeZone(),
|
||||
* the reference data is specified according to the Gregorian calendar, and the date
|
||||
* and time fields are local standard time.
|
||||
*
|
||||
* <p>Note: Don't call this method. Instead, call the getOffset(UDate...) overload,
|
||||
* which returns both the raw and the DST offset for a given time. This method
|
||||
* is retained only for backward compatibility.
|
||||
*
|
||||
* @param era The reference date's era
|
||||
* @param year The reference date's year
|
||||
* @param month The reference date's month (0-based; 0 is January)
|
||||
* @param day The reference date's day-in-month (1-based)
|
||||
* @param dayOfWeek The reference date's day-of-week (1-based; 1 is Sunday)
|
||||
* @param millis The reference date's milliseconds in day, local standard time
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @return The offset in milliseconds to add to GMT to get local time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Gets the time zone offset, for current date, modified in case of
|
||||
* daylight savings. This is the offset to add *to* UTC to get local time.
|
||||
*
|
||||
* <p>Note: Don't call this method. Instead, call the getOffset(UDate...) overload,
|
||||
* which returns both the raw and the DST offset for a given time. This method
|
||||
* is retained only for backward compatibility.
|
||||
*
|
||||
* @param era The reference date's era
|
||||
* @param year The reference date's year
|
||||
* @param month The reference date's month (0-based; 0 is January)
|
||||
* @param day The reference date's day-in-month (1-based)
|
||||
* @param dayOfWeek The reference date's day-of-week (1-based; 1 is Sunday)
|
||||
* @param millis The reference date's milliseconds in day, local standard time
|
||||
* @param monthLength The length of the given month in days.
|
||||
* @param status Output param to filled in with a success or an error.
|
||||
* @return The offset in milliseconds to add to GMT to get local time.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
|
||||
uint8_t dayOfWeek, int32_t millis,
|
||||
int32_t monthLength, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Returns the time zone raw and GMT offset for the given moment
|
||||
* in time. Upon return, local-millis = GMT-millis + rawOffset +
|
||||
* dstOffset. All computations are performed in the proleptic
|
||||
* Gregorian calendar. The default implementation in the TimeZone
|
||||
* class delegates to the 8-argument getOffset().
|
||||
*
|
||||
* @param date moment in time for which to return offsets, in
|
||||
* units of milliseconds from January 1, 1970 0:00 GMT, either GMT
|
||||
* time or local wall time, depending on `local'.
|
||||
* @param local if true, `date' is local wall time; otherwise it
|
||||
* is in GMT time.
|
||||
* @param rawOffset output parameter to receive the raw offset, that
|
||||
* is, the offset not including DST adjustments
|
||||
* @param dstOffset output parameter to receive the DST offset,
|
||||
* that is, the offset to be added to `rawOffset' to obtain the
|
||||
* total offset between local and GMT time. If DST is not in
|
||||
* effect, this value is zero; otherwise it is a positive value,
|
||||
* typically one hour.
|
||||
* @param ec input-output error code
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getOffset(UDate date, UBool local, int32_t& rawOffset,
|
||||
int32_t& dstOffset, UErrorCode& ec) const;
|
||||
|
||||
/**
|
||||
* Sets the TimeZone's raw GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time, before taking daylight savings time into account).
|
||||
*
|
||||
* @param offsetMillis The new raw GMT offset for this time zone.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void setRawOffset(int32_t offsetMillis);
|
||||
|
||||
/**
|
||||
* Returns the TimeZone's raw GMT offset (i.e., the number of milliseconds to add
|
||||
* to GMT to get local time, before taking daylight savings time into account).
|
||||
*
|
||||
* @return The TimeZone's raw GMT offset.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t getRawOffset(void) const;
|
||||
|
||||
/**
|
||||
* Queries if this time zone uses daylight savings time.
|
||||
* @return true if this time zone uses daylight savings time,
|
||||
* false, otherwise.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool useDaylightTime(void) const;
|
||||
|
||||
/**
|
||||
* Queries if the given date is in daylight savings time in
|
||||
* this time zone.
|
||||
* This method is wasteful since it creates a new GregorianCalendar and
|
||||
* deletes it each time it is called. This is a deprecated method
|
||||
* and provided only for Java compatibility.
|
||||
*
|
||||
* @param date the given UDate.
|
||||
* @param status Output param filled in with success/error code.
|
||||
* @return true if the given date is in daylight savings time,
|
||||
* false, otherwise.
|
||||
* @deprecated ICU 2.4. Use Calendar::inDaylightTime() instead.
|
||||
*/
|
||||
virtual UBool inDaylightTime(UDate date, UErrorCode& status) const;
|
||||
|
||||
/**
|
||||
* Returns true if this zone has the same rule and offset as another zone.
|
||||
* That is, if this zone differs only in ID, if at all.
|
||||
* @param other the <code>TimeZone</code> object to be compared with
|
||||
* @return true if the given zone is the same as this one,
|
||||
* with the possible exception of the ID
|
||||
* @stable ICU 3.8
|
||||
*/
|
||||
virtual UBool hasSameRules(const TimeZone& other) const;
|
||||
|
||||
/**
|
||||
* Gets the first time zone transition after the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the first transition after the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the most recent time zone transition before the base time.
|
||||
* @param base The base time.
|
||||
* @param inclusive Whether the base time is inclusive or not.
|
||||
* @param result Receives the most recent transition before the base time.
|
||||
* @return TRUE if the transition is found.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UBool getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/;
|
||||
|
||||
/**
|
||||
* Returns the number of <code>TimeZoneRule</code>s which represents time transitions,
|
||||
* for this time zone, that is, all <code>TimeZoneRule</code>s for this time zone except
|
||||
* <code>InitialTimeZoneRule</code>. The return value range is 0 or any positive value.
|
||||
* @param status Receives error status code.
|
||||
* @return The number of <code>TimeZoneRule</code>s representing time transitions.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual int32_t countTransitionRules(UErrorCode& status) /*const*/;
|
||||
|
||||
/**
|
||||
* Gets the <code>InitialTimeZoneRule</code> and the set of <code>TimeZoneRule</code>
|
||||
* which represent time transitions for this time zone. On successful return,
|
||||
* the argument initial points to non-NULL <code>InitialTimeZoneRule</code> and
|
||||
* the array trsrules is filled with 0 or multiple <code>TimeZoneRule</code>
|
||||
* instances up to the size specified by trscount. The results are referencing the
|
||||
* rule instance held by this time zone instance. Therefore, after this time zone
|
||||
* is destructed, they are no longer available.
|
||||
* @param initial Receives the initial timezone rule
|
||||
* @param trsrules Receives the timezone transition rules
|
||||
* @param trscount On input, specify the size of the array 'transitions' receiving
|
||||
* the timezone transition rules. On output, actual number of
|
||||
* rules filled in the array will be set.
|
||||
* @param status Receives error status code.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual void getTimeZoneRules(const InitialTimeZoneRule*& initial,
|
||||
const TimeZoneRule* trsrules[], int32_t& trscount, UErrorCode& status) /*const*/;
|
||||
|
||||
private:
|
||||
enum { DEFAULT_VTIMEZONE_LINES = 100 };
|
||||
|
||||
/**
|
||||
* Default constructor.
|
||||
*/
|
||||
VTimeZone();
|
||||
static VTimeZone* createVTimeZone(VTZReader* reader);
|
||||
void write(VTZWriter& writer, UErrorCode& status) const;
|
||||
void write(UDate start, VTZWriter& writer, UErrorCode& status) /*const*/;
|
||||
void writeSimple(UDate time, VTZWriter& writer, UErrorCode& status) /*const*/;
|
||||
void load(VTZReader& reader, UErrorCode& status);
|
||||
void parse(UErrorCode& status);
|
||||
|
||||
void writeZone(VTZWriter& w, BasicTimeZone& basictz, UVector* customProps,
|
||||
UErrorCode& status) const;
|
||||
|
||||
void writeHeaders(VTZWriter& w, UErrorCode& status) const;
|
||||
void writeFooter(VTZWriter& writer, UErrorCode& status) const;
|
||||
|
||||
void writeZonePropsByTime(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset, UDate time, UErrorCode& status) const;
|
||||
void writeZonePropsByDOM(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset,
|
||||
int32_t month, int32_t dayOfMonth, UDate startTime, UDate untilTime,
|
||||
UErrorCode& status) const;
|
||||
void writeZonePropsByDOW(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset,
|
||||
int32_t month, int32_t weekInMonth, int32_t dayOfWeek,
|
||||
UDate startTime, UDate untilTime, UErrorCode& status) const;
|
||||
void writeZonePropsByDOW_GEQ_DOM(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset,
|
||||
int32_t month, int32_t dayOfMonth, int32_t dayOfWeek,
|
||||
UDate startTime, UDate untilTime, UErrorCode& status) const;
|
||||
void writeZonePropsByDOW_GEQ_DOM_sub(VTZWriter& writer, int32_t month, int32_t dayOfMonth,
|
||||
int32_t dayOfWeek, int32_t numDays,
|
||||
UDate untilTime, int32_t fromOffset, UErrorCode& status) const;
|
||||
void writeZonePropsByDOW_LEQ_DOM(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset,
|
||||
int32_t month, int32_t dayOfMonth, int32_t dayOfWeek,
|
||||
UDate startTime, UDate untilTime, UErrorCode& status) const;
|
||||
void writeFinalRule(VTZWriter& writer, UBool isDst, const AnnualTimeZoneRule* rule,
|
||||
int32_t fromRawOffset, int32_t fromDSTSavings,
|
||||
UDate startTime, UErrorCode& status) const;
|
||||
|
||||
void beginZoneProps(VTZWriter& writer, UBool isDst, const UnicodeString& tzname,
|
||||
int32_t fromOffset, int32_t toOffset, UDate startTime, UErrorCode& status) const;
|
||||
void endZoneProps(VTZWriter& writer, UBool isDst, UErrorCode& status) const;
|
||||
void beginRRULE(VTZWriter& writer, int32_t month, UErrorCode& status) const;
|
||||
void appendUNTIL(VTZWriter& writer, const UnicodeString& until, UErrorCode& status) const;
|
||||
|
||||
BasicTimeZone *tz;
|
||||
UVector *vtzlines;
|
||||
UnicodeString tzurl;
|
||||
UDate lastmod;
|
||||
UnicodeString olsonzid;
|
||||
UnicodeString icutzver;
|
||||
|
||||
public:
|
||||
/**
|
||||
* Return the class ID for this class. This is useful only for comparing to
|
||||
* a return value from getDynamicClassID(). For example:
|
||||
* <pre>
|
||||
* . Base* polymorphic_pointer = createPolymorphicObject();
|
||||
* . if (polymorphic_pointer->getDynamicClassID() ==
|
||||
* . erived::getStaticClassID()) ...
|
||||
* </pre>
|
||||
* @return The class ID for all objects of this class.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
static UClassID U_EXPORT2 getStaticClassID(void);
|
||||
|
||||
/**
|
||||
* Returns a unique class ID POLYMORPHICALLY. Pure virtual override. This
|
||||
* method is to implement a simple version of RTTI, since not all C++
|
||||
* compilers support genuine RTTI. Polymorphic operator==() and clone()
|
||||
* methods call this method.
|
||||
*
|
||||
* @return The class ID for this object. All objects of a
|
||||
* given class have the same class ID. Objects of
|
||||
* other classes have different class IDs.
|
||||
* @draft ICU 3.8
|
||||
*/
|
||||
virtual UClassID getDynamicClassID(void) const;
|
||||
};
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // VTZONE_H
|
||||
//eof
|
2458
icu4c/source/i18n/vtzone.cpp
Normal file
2458
icu4c/source/i18n/vtzone.cpp
Normal file
File diff suppressed because it is too large
Load diff
|
@ -56,7 +56,7 @@ jamotest.o srchtest.o reptest.o regextst.o \
|
|||
itrbnf.o itrbnfrt.o itrbnfp.o ucaconf.o icusvtst.o \
|
||||
uobjtest.o idnaref.o idnaconf.o nptrans.o punyref.o testidn.o testidna.o incaltst.o \
|
||||
calcasts.o v32test.o uvectest.o textfile.o tokiter.o utxttest.o \
|
||||
windttst.o winnmtst.o winutil.o csdetest.o
|
||||
windttst.o winnmtst.o winutil.o csdetest.o tzrulets.o
|
||||
|
||||
|
||||
DEPS = $(OBJECTS:.o=.d)
|
||||
|
|
|
@ -764,6 +764,14 @@
|
|||
RelativePath=".\tzregts.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\tzrulets.cpp"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\tzrulets.h"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\tztest.cpp"
|
||||
>
|
||||
|
|
|
@ -42,6 +42,7 @@
|
|||
#include "astrotst.h" // AstroTest
|
||||
#include "incaltst.h" // IntlCalendarTest
|
||||
#include "calcasts.h" // CalendarCaseTest
|
||||
#include "tzrulets.h" // TimeZoneRuleTest
|
||||
|
||||
#define TESTCLASS(id, TestClass) \
|
||||
case id: \
|
||||
|
@ -101,6 +102,7 @@ void IntlTestFormat::runIndexedTest( int32_t index, UBool exec, const char* &nam
|
|||
TESTCLASS(26,IntlCalendarTest);
|
||||
TESTCLASS(27,AstroTest);
|
||||
TESTCLASS(28,CalendarCaseTest);
|
||||
TESTCLASS(29,TimeZoneRuleTest);
|
||||
|
||||
default: name = ""; break; //needed to end loop
|
||||
}
|
||||
|
|
1125
icu4c/source/test/intltest/tzrulets.cpp
Normal file
1125
icu4c/source/test/intltest/tzrulets.cpp
Normal file
File diff suppressed because it is too large
Load diff
46
icu4c/source/test/intltest/tzrulets.h
Normal file
46
icu4c/source/test/intltest/tzrulets.h
Normal file
|
@ -0,0 +1,46 @@
|
|||
/*
|
||||
*******************************************************************************
|
||||
* Copyright (C) 2007, International Business Machines Corporation and *
|
||||
* others. All Rights Reserved. *
|
||||
*******************************************************************************
|
||||
*/
|
||||
|
||||
#ifndef _TIMEZONERULETEST_
|
||||
#define _TIMEZONERULETEST_
|
||||
|
||||
#include "unicode/utypes.h"
|
||||
#include "caltztst.h"
|
||||
|
||||
#if !UCONFIG_NO_FORMATTING
|
||||
|
||||
/**
|
||||
* Tests for TimeZoneRule, RuleBasedTimeZone and VTimeZone
|
||||
*/
|
||||
class TimeZoneRuleTest : public CalendarTimeZoneTest {
|
||||
// IntlTest override
|
||||
void runIndexedTest(int32_t index, UBool exec, const char*& name, char* par);
|
||||
public:
|
||||
void TestSimpleRuleBasedTimeZone(void);
|
||||
void TestHistoricalRuleBasedTimeZone(void);
|
||||
void TestOlsonTransition(void);
|
||||
void TestRBTZTransition(void);
|
||||
void TestHasEquivalentTransitions(void);
|
||||
void TestVTimeZoneRoundTrip(void);
|
||||
void TestVTimeZoneRoundTripPartial(void);
|
||||
void TestVTimeZoneSimpleWrite(void);
|
||||
void TestVTimeZoneHeaderProps(void);
|
||||
void TestGetSimpleRules(void);
|
||||
|
||||
private:
|
||||
void verifyTransitions(BasicTimeZone& icutz, UDate start, UDate end);
|
||||
void compareTransitionsAscending(BasicTimeZone& z1, BasicTimeZone& z2,
|
||||
UDate start, UDate end, UBool inclusive);
|
||||
void compareTransitionsDescending(BasicTimeZone& z1, BasicTimeZone& z2,
|
||||
UDate start, UDate end, UBool inclusive);
|
||||
UDate getUTCMillis(int32_t year, int32_t month, int32_t dom,
|
||||
int32_t hour=0, int32_t min=0, int32_t sec=0, int32_t msec=0);
|
||||
};
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
||||
#endif // _TIMEZONERULETEST_
|
Loading…
Add table
Reference in a new issue