/*
*******************************************************************************
-* Copyright (C) 2003, International Business Machines Corporation and *
+* Copyright (C) 2003-2013, International Business Machines Corporation and *
* others. All Rights Reserved. *
*******************************************************************************
*
#include "buddhcal.h"
#include "unicode/gregocal.h"
-#include "mutex.h"
+#include "umutex.h"
#include <float.h>
U_NAMESPACE_BEGIN
-const char BuddhistCalendar::fgClassID = 0; // Value is irrelevant
+UOBJECT_DEFINE_RTTI_IMPLEMENTATION(BuddhistCalendar)
-static const int32_t kMaxEra = 0; // only 1 era
+//static const int32_t kMaxEra = 0; // only 1 era
static const int32_t kBuddhistEraStart = -543; // 544 BC (Gregorian)
-static const int32_t kGregorianEpoch = 1970;
+static const int32_t kGregorianEpoch = 1970; // used as the default value of EXTENDED_YEAR
BuddhistCalendar::BuddhistCalendar(const Locale& aLocale, UErrorCode& success)
- : GregorianCalendar(aLocale, success)
+: GregorianCalendar(aLocale, success)
{
setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.
}
}
BuddhistCalendar::BuddhistCalendar(const BuddhistCalendar& source)
- : GregorianCalendar(source)
+: GregorianCalendar(source)
{
}
BuddhistCalendar& BuddhistCalendar::operator= ( const BuddhistCalendar& right)
{
- GregorianCalendar::operator=(right);
- return *this;
+ GregorianCalendar::operator=(right);
+ return *this;
}
Calendar* BuddhistCalendar::clone(void) const
{
- return new BuddhistCalendar(*this);
+ return new BuddhistCalendar(*this);
}
const char *BuddhistCalendar::getType() const
{
- return "buddhist";
+ return "buddhist";
}
-int32_t
-BuddhistCalendar::getMaximum(UCalendarDateFields field) const
+int32_t BuddhistCalendar::handleGetExtendedYear()
{
- if(field == UCAL_ERA) {
- return kMaxEra;
- } else {
- return GregorianCalendar::getMaximum(field);
- }
-}
-
-int32_t
-BuddhistCalendar::getLeastMaximum(UCalendarDateFields field) const
-{
- if(field == UCAL_ERA) {
- return kMaxEra;
- } else {
- return GregorianCalendar::getLeastMaximum(field);
- }
-}
-
-int32_t
-BuddhistCalendar::monthLength(int32_t month, int32_t year) const
-{
- return GregorianCalendar::monthLength(month,year);
+ // EXTENDED_YEAR in BuddhistCalendar is a Gregorian year.
+ // The default value of EXTENDED_YEAR is 1970 (Buddhist 2513)
+ int32_t year;
+ if (newerField(UCAL_EXTENDED_YEAR, UCAL_YEAR) == UCAL_EXTENDED_YEAR) {
+ year = internalGet(UCAL_EXTENDED_YEAR, kGregorianEpoch);
+ } else {
+ // extended year is a gregorian year, where 1 = 1AD, 0 = 1BC, -1 = 2BC, etc
+ year = internalGet(UCAL_YEAR, kGregorianEpoch - kBuddhistEraStart)
+ + kBuddhistEraStart;
+ }
+ return year;
}
+int32_t BuddhistCalendar::handleComputeMonthStart(int32_t eyear, int32_t month,
-int32_t
-BuddhistCalendar::monthLength(int32_t month) const
+ UBool useMonth) const
{
- UErrorCode status = U_ZERO_ERROR;
- int32_t year = internalGet(UCAL_YEAR);
- // ignore era
- return GregorianCalendar::monthLength(month, getGregorianYear(status));
+ return GregorianCalendar::handleComputeMonthStart(eyear, month, useMonth);
}
-int32_t BuddhistCalendar::internalGetEra() const
+void BuddhistCalendar::handleComputeFields(int32_t julianDay, UErrorCode& status)
{
- return isSet(UCAL_ERA) ? internalGet(UCAL_ERA) : BE;
+ GregorianCalendar::handleComputeFields(julianDay, status);
+ int32_t y = internalGet(UCAL_EXTENDED_YEAR) - kBuddhistEraStart;
+ internalSet(UCAL_ERA, 0);
+ internalSet(UCAL_YEAR, y);
}
-int32_t
-BuddhistCalendar::getGregorianYear(UErrorCode &status) const
+int32_t BuddhistCalendar::handleGetLimit(UCalendarDateFields field, ELimitType limitType) const
{
- int32_t year = (fStamp[UCAL_YEAR] != kUnset) ? internalGet(UCAL_YEAR) : kGregorianEpoch+kBuddhistEraStart;
- int32_t era = BE;
- if (fStamp[UCAL_ERA] != kUnset) {
- era = internalGet(UCAL_ERA);
- if (era != BE) {
- status = U_ILLEGAL_ARGUMENT_ERROR;
- return kGregorianEpoch + kBuddhistEraStart;
+ if(field == UCAL_ERA) {
+ return BE;
+ } else {
+ return GregorianCalendar::handleGetLimit(field,limitType);
}
- }
- return year + kBuddhistEraStart;
}
+#if 0
void BuddhistCalendar::timeToFields(UDate theTime, UBool quick, UErrorCode& status)
{
- GregorianCalendar::timeToFields(theTime, quick, status);
-
- int32_t era = internalGet(UCAL_ERA);
- int32_t year = internalGet(UCAL_YEAR);
-
- if(era == GregorianCalendar::BC) {
- year = 1-year;
- era = BuddhistCalendar::BE;
- } else if(era == GregorianCalendar::AD) {
- era = BuddhistCalendar::BE;
- } else {
- status = U_INTERNAL_PROGRAM_ERROR;
- }
-
- year = year - kBuddhistEraStart;
-
- internalSet(UCAL_ERA, era);
- internalSet(UCAL_YEAR, year);
-}
+ //Calendar::timeToFields(theTime, quick, status);
-void BuddhistCalendar::add(UCalendarDateFields field, int32_t amount, UErrorCode& status)
-{
- if (U_FAILURE(status))
- return;
-
- if (amount == 0)
- return; // Do nothing!
-
- if(field == UCAL_YEAR /* || field == UCAL_YEAR_WOY */) {
- int32_t year = internalGet(field);
- int32_t era = internalGetEra();
-
- year += amount;
-
- set(field,year);
- pinDayOfMonth();
+ int32_t era = internalGet(UCAL_ERA);
+ int32_t year = internalGet(UCAL_YEAR);
+
+ if(era == GregorianCalendar::BC) {
+ year = 1-year;
+ era = BuddhistCalendar::BE;
+ } else if(era == GregorianCalendar::AD) {
+ era = BuddhistCalendar::BE;
} else {
- GregorianCalendar::add(field,amount,status);
+ status = U_INTERNAL_PROGRAM_ERROR;
}
-}
-
+ year = year - kBuddhistEraStart;
-// default century
-const UDate BuddhistCalendar::fgSystemDefaultCentury = DBL_MIN;
-const int32_t BuddhistCalendar::fgSystemDefaultCenturyYear = -1;
+ internalSet(UCAL_ERA, era);
+ internalSet(UCAL_YEAR, year);
+}
+#endif
-UDate BuddhistCalendar::fgSystemDefaultCenturyStart = DBL_MIN;
-int32_t BuddhistCalendar::fgSystemDefaultCenturyStartYear = -1;
+/**
+ * The system maintains a static default century start date. This is initialized
+ * the first time it is used. Once the system default century date and year
+ * are set, they do not change.
+ */
+static UDate gSystemDefaultCenturyStart = DBL_MIN;
+static int32_t gSystemDefaultCenturyStartYear = -1;
+static icu::UInitOnce gBCInitOnce;
UBool BuddhistCalendar::haveDefaultCentury() const
{
- return TRUE;
+ return TRUE;
}
-UDate BuddhistCalendar::defaultCenturyStart() const
-{
- return internalGetDefaultCenturyStart();
-}
-
-int32_t BuddhistCalendar::defaultCenturyStartYear() const
+static void U_CALLCONV
+initializeSystemDefaultCentury()
{
- return internalGetDefaultCenturyStartYear();
-}
-
-UDate
-BuddhistCalendar::internalGetDefaultCenturyStart() const
-{
- // lazy-evaluate systemDefaultCenturyStart
- UBool needsUpdate;
- {
- Mutex m;
- needsUpdate = (fgSystemDefaultCenturyStart == fgSystemDefaultCentury);
- }
-
- if (needsUpdate) {
- initializeSystemDefaultCentury();
- }
-
- // use defaultCenturyStart unless it's the flag value;
- // then use systemDefaultCenturyStart
-
- return fgSystemDefaultCenturyStart;
+ // initialize systemDefaultCentury and systemDefaultCenturyYear based
+ // on the current time. They'll be set to 80 years before
+ // the current time.
+ UErrorCode status = U_ZERO_ERROR;
+ BuddhistCalendar calendar(Locale("@calendar=buddhist"),status);
+ if (U_SUCCESS(status)) {
+ calendar.setTime(Calendar::getNow(), status);
+ calendar.add(UCAL_YEAR, -80, status);
+ UDate newStart = calendar.getTime(status);
+ int32_t newYear = calendar.get(UCAL_YEAR, status);
+ gSystemDefaultCenturyStartYear = newYear;
+ gSystemDefaultCenturyStart = newStart;
+ }
+ // We have no recourse upon failure unless we want to propagate the failure
+ // out.
}
-int32_t
-BuddhistCalendar::internalGetDefaultCenturyStartYear() const
+UDate BuddhistCalendar::defaultCenturyStart() const
{
- // lazy-evaluate systemDefaultCenturyStartYear
- UBool needsUpdate;
- {
- Mutex m;
- needsUpdate = (fgSystemDefaultCenturyStart == fgSystemDefaultCentury);
- }
-
- if (needsUpdate) {
- initializeSystemDefaultCentury();
- }
-
- // use defaultCenturyStart unless it's the flag value;
- // then use systemDefaultCenturyStartYear
-
- return fgSystemDefaultCenturyStartYear;
+ // lazy-evaluate systemDefaultCenturyStart and systemDefaultCenturyStartYear
+ umtx_initOnce(gBCInitOnce, &initializeSystemDefaultCentury);
+ return gSystemDefaultCenturyStart;
}
-void
-BuddhistCalendar::initializeSystemDefaultCentury()
+int32_t BuddhistCalendar::defaultCenturyStartYear() const
{
- // initialize systemDefaultCentury and systemDefaultCenturyYear based
- // on the current time. They'll be set to 80 years before
- // the current time.
- // No point in locking as it should be idempotent.
- if (fgSystemDefaultCenturyStart == fgSystemDefaultCentury)
- {
- UErrorCode status = U_ZERO_ERROR;
- Calendar *calendar = new BuddhistCalendar(Locale("th_TH_TRADITIONAL"),status);
- if (calendar != NULL && U_SUCCESS(status))
- {
- calendar->setTime(Calendar::getNow(), status);
- calendar->add(UCAL_YEAR, -80, status);
- UDate newStart = calendar->getTime(status);
- int32_t newYear = calendar->get(UCAL_YEAR, status);
- {
- Mutex m;
- fgSystemDefaultCenturyStart = newStart;
- fgSystemDefaultCenturyStartYear = newYear;
- }
- delete calendar;
- }
- // We have no recourse upon failure unless we want to propagate the failure
- // out.
- }
+ // lazy-evaluate systemDefaultCenturyStartYear and systemDefaultCenturyStart
+ umtx_initOnce(gBCInitOnce, &initializeSystemDefaultCentury);
+ return gSystemDefaultCenturyStartYear;
}