// Copyright (C) 2016 and later: Unicode, Inc. and others.
// License & terms of use: http://www.unicode.org/copyright.html
/*
*******************************************************************************
* Copyright (C) 2007-2012, International Business Machines Corporation and
* others. All Rights Reserved.
*******************************************************************************
*/
#include "utypeinfo.h" // for 'typeid' to work
#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_CDECL_BEGIN
// UComparator function for sorting start times
static int32_t U_CALLCONV
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;
}
U_CDECL_END
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) ||
(typeid(*this) == typeid(that) &&
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) ||
(typeid(*this) == typeid(other) &&
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 TimeZoneRule& that) const {
return ((this == &that) ||
(typeid(*this) == typeid(that) &&
TimeZoneRule::operator==(that)));
}
UBool
InitialTimeZoneRule::operator!=(const TimeZoneRule& that) const {
return !operator==(that);
}
UBool
InitialTimeZoneRule::isEquivalentTo(const TimeZoneRule& other) const {
if (this == &other) {
return TRUE;
}
if (typeid(*this) != typeid(other) || 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(UDate /*base*/,
int32_t /*prevRawOffset*/,
int32_t /*prevDSTSavings*/,
UBool /*inclusive*/,
UDate& /*result*/) const {
return FALSE;
}
UBool
InitialTimeZoneRule::getPreviousStart(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 TimeZoneRule& that) const {
if (this == &that) {
return TRUE;
}
if (typeid(*this) != typeid(that)) {
return FALSE;
}
AnnualTimeZoneRule *atzr = (AnnualTimeZoneRule*)&that;
return (*fDateTimeRule == *(atzr->fDateTimeRule) &&
fStartYear == atzr->fStartYear &&
fEndYear == atzr->fEndYear);
}
UBool
AnnualTimeZoneRule::operator!=(const TimeZoneRule& 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 (typeid(*this) != typeid(other) || 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(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(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)
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 TimeZoneRule& that) const {
if (this == &that) {
return TRUE;
}
if (typeid(*this) != typeid(that) || TimeZoneRule::operator==(that) == FALSE) {
return FALSE;
}
TimeArrayTimeZoneRule *tatzr = (TimeArrayTimeZoneRule*)&that;
if (fTimeRuleType != tatzr->fTimeRuleType ||
fNumStartTimes != tatzr->fNumStartTimes) {
return FALSE;
}
// Compare start times
UBool res = TRUE;
for (int32_t i = 0; i < fNumStartTimes; i++) {
if (fStartTimes[i] != tatzr->fStartTimes[i]) {
res = FALSE;
break;
}
}
return res;
}
UBool
TimeArrayTimeZoneRule::operator!=(const TimeZoneRule& 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 (typeid(*this) != typeid(other) || 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(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(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