1/* 2 ******************************************************************************* 3 * Copyright (C) 2003-2008, International Business Machines Corporation and * 4 * others. All Rights Reserved. * 5 ******************************************************************************* 6 * 7 * File TAIWNCAL.CPP 8 * 9 * Modification History: 10 * 05/13/2003 srl copied from gregocal.cpp 11 * 06/29/2007 srl copied from buddhcal.cpp 12 * 05/12/2008 jce modified to use calendar=roc per CLDR 13 * 14 */ 15 16#include "unicode/utypes.h" 17 18#if !UCONFIG_NO_FORMATTING 19 20#include "taiwncal.h" 21#include "unicode/gregocal.h" 22#include "umutex.h" 23#include <float.h> 24 25U_NAMESPACE_BEGIN 26 27UOBJECT_DEFINE_RTTI_IMPLEMENTATION(TaiwanCalendar) 28 29static const int32_t kTaiwanEraStart = 1911; // 1911 (Gregorian) 30 31static const int32_t kGregorianEpoch = 1970; 32 33TaiwanCalendar::TaiwanCalendar(const Locale& aLocale, UErrorCode& success) 34: GregorianCalendar(aLocale, success) 35{ 36 setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly. 37} 38 39TaiwanCalendar::~TaiwanCalendar() 40{ 41} 42 43TaiwanCalendar::TaiwanCalendar(const TaiwanCalendar& source) 44: GregorianCalendar(source) 45{ 46} 47 48TaiwanCalendar& TaiwanCalendar::operator= ( const TaiwanCalendar& right) 49{ 50 GregorianCalendar::operator=(right); 51 return *this; 52} 53 54Calendar* TaiwanCalendar::clone(void) const 55{ 56 return new TaiwanCalendar(*this); 57} 58 59const char *TaiwanCalendar::getType() const 60{ 61 return "roc"; 62} 63 64int32_t TaiwanCalendar::handleGetExtendedYear() 65{ 66 // EXTENDED_YEAR in TaiwanCalendar is a Gregorian year 67 // The default value of EXTENDED_YEAR is 1970 (Minguo 59) 68 int32_t year = kGregorianEpoch; 69 70 if (newerField(UCAL_EXTENDED_YEAR, UCAL_YEAR) == UCAL_EXTENDED_YEAR 71 && newerField(UCAL_EXTENDED_YEAR, UCAL_ERA) == UCAL_EXTENDED_YEAR) { 72 year = internalGet(UCAL_EXTENDED_YEAR, kGregorianEpoch); 73 } else { 74 int32_t era = internalGet(UCAL_ERA, MINGUO); 75 if(era == MINGUO) { 76 year = internalGet(UCAL_YEAR, 1) + kTaiwanEraStart; 77 } else if(era == BEFORE_MINGUO) { 78 year = 1 - internalGet(UCAL_YEAR, 1) + kTaiwanEraStart; 79 } 80 } 81 return year; 82} 83 84void TaiwanCalendar::handleComputeFields(int32_t julianDay, UErrorCode& status) 85{ 86 GregorianCalendar::handleComputeFields(julianDay, status); 87 int32_t y = internalGet(UCAL_EXTENDED_YEAR) - kTaiwanEraStart; 88 if(y>0) { 89 internalSet(UCAL_ERA, MINGUO); 90 internalSet(UCAL_YEAR, y); 91 } else { 92 internalSet(UCAL_ERA, BEFORE_MINGUO); 93 internalSet(UCAL_YEAR, 1-y); 94 } 95} 96 97int32_t TaiwanCalendar::handleGetLimit(UCalendarDateFields field, ELimitType limitType) const 98{ 99 if(field == UCAL_ERA) { 100 if(limitType == UCAL_LIMIT_MINIMUM || limitType == UCAL_LIMIT_GREATEST_MINIMUM) { 101 return BEFORE_MINGUO; 102 } else { 103 return MINGUO; 104 } 105 } else { 106 return GregorianCalendar::handleGetLimit(field,limitType); 107 } 108} 109 110#if 0 111void TaiwanCalendar::timeToFields(UDate theTime, UBool quick, UErrorCode& status) 112{ 113 //Calendar::timeToFields(theTime, quick, status); 114 115 int32_t era = internalGet(UCAL_ERA); 116 int32_t year = internalGet(UCAL_YEAR); 117 118 if(era == GregorianCalendar::BC) { 119 year = 1-year; 120 era = TaiwanCalendar::MINGUO; 121 } else if(era == GregorianCalendar::AD) { 122 era = TaiwanCalendar::MINGUO; 123 } else { 124 status = U_INTERNAL_PROGRAM_ERROR; 125 } 126 127 year = year - kTaiwanEraStart; 128 129 internalSet(UCAL_ERA, era); 130 internalSet(UCAL_YEAR, year); 131} 132#endif 133 134// default century 135const UDate TaiwanCalendar::fgSystemDefaultCentury = DBL_MIN; 136const int32_t TaiwanCalendar::fgSystemDefaultCenturyYear = -1; 137 138UDate TaiwanCalendar::fgSystemDefaultCenturyStart = DBL_MIN; 139int32_t TaiwanCalendar::fgSystemDefaultCenturyStartYear = -1; 140 141 142UBool TaiwanCalendar::haveDefaultCentury() const 143{ 144 return TRUE; 145} 146 147UDate TaiwanCalendar::defaultCenturyStart() const 148{ 149 return internalGetDefaultCenturyStart(); 150} 151 152int32_t TaiwanCalendar::defaultCenturyStartYear() const 153{ 154 return internalGetDefaultCenturyStartYear(); 155} 156 157UDate 158TaiwanCalendar::internalGetDefaultCenturyStart() const 159{ 160 // lazy-evaluate systemDefaultCenturyStart 161 UBool needsUpdate; 162 UMTX_CHECK(NULL, (fgSystemDefaultCenturyStart == fgSystemDefaultCentury), needsUpdate); 163 164 if (needsUpdate) { 165 initializeSystemDefaultCentury(); 166 } 167 168 // use defaultCenturyStart unless it's the flag value; 169 // then use systemDefaultCenturyStart 170 171 return fgSystemDefaultCenturyStart; 172} 173 174int32_t 175TaiwanCalendar::internalGetDefaultCenturyStartYear() const 176{ 177 // lazy-evaluate systemDefaultCenturyStartYear 178 UBool needsUpdate; 179 UMTX_CHECK(NULL, (fgSystemDefaultCenturyStart == fgSystemDefaultCentury), needsUpdate); 180 181 if (needsUpdate) { 182 initializeSystemDefaultCentury(); 183 } 184 185 // use defaultCenturyStart unless it's the flag value; 186 // then use systemDefaultCenturyStartYear 187 188 return fgSystemDefaultCenturyStartYear; 189} 190 191void 192TaiwanCalendar::initializeSystemDefaultCentury() 193{ 194 // initialize systemDefaultCentury and systemDefaultCenturyYear based 195 // on the current time. They'll be set to 80 years before 196 // the current time. 197 UErrorCode status = U_ZERO_ERROR; 198 TaiwanCalendar calendar(Locale("@calendar=roc"),status); 199 if (U_SUCCESS(status)) 200 { 201 calendar.setTime(Calendar::getNow(), status); 202 calendar.add(UCAL_YEAR, -80, status); 203 UDate newStart = calendar.getTime(status); 204 int32_t newYear = calendar.get(UCAL_YEAR, status); 205 umtx_lock(NULL); 206 if (fgSystemDefaultCenturyStart == fgSystemDefaultCentury) 207 { 208 fgSystemDefaultCenturyStartYear = newYear; 209 fgSystemDefaultCenturyStart = newStart; 210 } 211 umtx_unlock(NULL); 212 } 213 // We have no recourse upon failure unless we want to propagate the failure 214 // out. 215} 216 217 218U_NAMESPACE_END 219 220#endif 221