bfe100bab8
X-SVN-Rev: 38963
2106 lines
82 KiB
C++
2106 lines
82 KiB
C++
// Copyright (C) 2016 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
/*
|
|
*******************************************************************************
|
|
* Copyright (C) 1997-2016, International Business Machines Corporation and *
|
|
* others. All Rights Reserved. *
|
|
*******************************************************************************
|
|
*
|
|
* File DTFMTSYM.CPP
|
|
*
|
|
* Modification History:
|
|
*
|
|
* Date Name Description
|
|
* 02/19/97 aliu Converted from java.
|
|
* 07/21/98 stephen Added getZoneIndex
|
|
* Changed weekdays/short weekdays to be one-based
|
|
* 06/14/99 stephen Removed SimpleDateFormat::fgTimeZoneDataSuffix
|
|
* 11/16/99 weiv Added 'Y' and 'e' to fgPatternChars
|
|
* 03/27/00 weiv Keeping resource bundle around!
|
|
* 06/30/05 emmons Added eraNames, narrow month/day, standalone context
|
|
* 10/12/05 emmons Added setters for eraNames, month/day by width/context
|
|
*******************************************************************************
|
|
*/
|
|
#include "unicode/utypes.h"
|
|
|
|
#if !UCONFIG_NO_FORMATTING
|
|
#include "unicode/ustring.h"
|
|
#include "unicode/localpointer.h"
|
|
#include "unicode/dtfmtsym.h"
|
|
#include "unicode/smpdtfmt.h"
|
|
#include "unicode/msgfmt.h"
|
|
#include "unicode/numsys.h"
|
|
#include "unicode/tznames.h"
|
|
#include "cpputils.h"
|
|
#include "umutex.h"
|
|
#include "cmemory.h"
|
|
#include "cstring.h"
|
|
#include "locbased.h"
|
|
#include "gregoimp.h"
|
|
#include "hash.h"
|
|
#include "uresimp.h"
|
|
#include "ureslocs.h"
|
|
#include "shareddateformatsymbols.h"
|
|
#include "unicode/calendar.h"
|
|
#include "unifiedcache.h"
|
|
|
|
// *****************************************************************************
|
|
// class DateFormatSymbols
|
|
// *****************************************************************************
|
|
|
|
/**
|
|
* These are static arrays we use only in the case where we have no
|
|
* resource data.
|
|
*/
|
|
|
|
#if UDAT_HAS_PATTERN_CHAR_FOR_TIME_SEPARATOR
|
|
#define PATTERN_CHARS_LEN 38
|
|
#else
|
|
#define PATTERN_CHARS_LEN 37
|
|
#endif
|
|
|
|
/**
|
|
* Unlocalized date-time pattern characters. For example: 'y', 'd', etc. All
|
|
* locales use the same these unlocalized pattern characters.
|
|
*/
|
|
static const UChar gPatternChars[] = {
|
|
// if UDAT_HAS_PATTERN_CHAR_FOR_TIME_SEPARATOR:
|
|
// GyMdkHmsSEDFwWahKzYeugAZvcLQqVUOXxrbB:
|
|
// else:
|
|
// GyMdkHmsSEDFwWahKzYeugAZvcLQqVUOXxrbB
|
|
|
|
0x47, 0x79, 0x4D, 0x64, 0x6B, 0x48, 0x6D, 0x73, 0x53, 0x45,
|
|
0x44, 0x46, 0x77, 0x57, 0x61, 0x68, 0x4B, 0x7A, 0x59, 0x65,
|
|
0x75, 0x67, 0x41, 0x5A, 0x76, 0x63, 0x4c, 0x51, 0x71, 0x56,
|
|
0x55, 0x4F, 0x58, 0x78, 0x72, 0x62, 0x42,
|
|
#if UDAT_HAS_PATTERN_CHAR_FOR_TIME_SEPARATOR
|
|
0x3a,
|
|
#endif
|
|
0
|
|
};
|
|
|
|
//------------------------------------------------------
|
|
// Strings of last resort. These are only used if we have no resource
|
|
// files. They aren't designed for actual use, just for backup.
|
|
|
|
// These are the month names and abbreviations of last resort.
|
|
static const UChar gLastResortMonthNames[13][3] =
|
|
{
|
|
{0x0030, 0x0031, 0x0000}, /* "01" */
|
|
{0x0030, 0x0032, 0x0000}, /* "02" */
|
|
{0x0030, 0x0033, 0x0000}, /* "03" */
|
|
{0x0030, 0x0034, 0x0000}, /* "04" */
|
|
{0x0030, 0x0035, 0x0000}, /* "05" */
|
|
{0x0030, 0x0036, 0x0000}, /* "06" */
|
|
{0x0030, 0x0037, 0x0000}, /* "07" */
|
|
{0x0030, 0x0038, 0x0000}, /* "08" */
|
|
{0x0030, 0x0039, 0x0000}, /* "09" */
|
|
{0x0031, 0x0030, 0x0000}, /* "10" */
|
|
{0x0031, 0x0031, 0x0000}, /* "11" */
|
|
{0x0031, 0x0032, 0x0000}, /* "12" */
|
|
{0x0031, 0x0033, 0x0000} /* "13" */
|
|
};
|
|
|
|
// These are the weekday names and abbreviations of last resort.
|
|
static const UChar gLastResortDayNames[8][2] =
|
|
{
|
|
{0x0030, 0x0000}, /* "0" */
|
|
{0x0031, 0x0000}, /* "1" */
|
|
{0x0032, 0x0000}, /* "2" */
|
|
{0x0033, 0x0000}, /* "3" */
|
|
{0x0034, 0x0000}, /* "4" */
|
|
{0x0035, 0x0000}, /* "5" */
|
|
{0x0036, 0x0000}, /* "6" */
|
|
{0x0037, 0x0000} /* "7" */
|
|
};
|
|
|
|
// These are the quarter names and abbreviations of last resort.
|
|
static const UChar gLastResortQuarters[4][2] =
|
|
{
|
|
{0x0031, 0x0000}, /* "1" */
|
|
{0x0032, 0x0000}, /* "2" */
|
|
{0x0033, 0x0000}, /* "3" */
|
|
{0x0034, 0x0000}, /* "4" */
|
|
};
|
|
|
|
// These are the am/pm and BC/AD markers of last resort.
|
|
static const UChar gLastResortAmPmMarkers[2][3] =
|
|
{
|
|
{0x0041, 0x004D, 0x0000}, /* "AM" */
|
|
{0x0050, 0x004D, 0x0000} /* "PM" */
|
|
};
|
|
|
|
static const UChar gLastResortEras[2][3] =
|
|
{
|
|
{0x0042, 0x0043, 0x0000}, /* "BC" */
|
|
{0x0041, 0x0044, 0x0000} /* "AD" */
|
|
};
|
|
|
|
/* Sizes for the last resort string arrays */
|
|
typedef enum LastResortSize {
|
|
kMonthNum = 13,
|
|
kMonthLen = 3,
|
|
|
|
kDayNum = 8,
|
|
kDayLen = 2,
|
|
|
|
kAmPmNum = 2,
|
|
kAmPmLen = 3,
|
|
|
|
kQuarterNum = 4,
|
|
kQuarterLen = 2,
|
|
|
|
kEraNum = 2,
|
|
kEraLen = 3,
|
|
|
|
kZoneNum = 5,
|
|
kZoneLen = 4,
|
|
|
|
kGmtHourNum = 4,
|
|
kGmtHourLen = 10
|
|
} LastResortSize;
|
|
|
|
U_NAMESPACE_BEGIN
|
|
|
|
SharedDateFormatSymbols::~SharedDateFormatSymbols() {
|
|
}
|
|
|
|
template<> U_I18N_API
|
|
const SharedDateFormatSymbols *
|
|
LocaleCacheKey<SharedDateFormatSymbols>::createObject(
|
|
const void * /*unusedContext*/, UErrorCode &status) const {
|
|
char type[256];
|
|
Calendar::getCalendarTypeFromLocale(fLoc, type, UPRV_LENGTHOF(type), status);
|
|
if (U_FAILURE(status)) {
|
|
return NULL;
|
|
}
|
|
SharedDateFormatSymbols *shared
|
|
= new SharedDateFormatSymbols(fLoc, type, status);
|
|
if (shared == NULL) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
return NULL;
|
|
}
|
|
if (U_FAILURE(status)) {
|
|
delete shared;
|
|
return NULL;
|
|
}
|
|
shared->addRef();
|
|
return shared;
|
|
}
|
|
|
|
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(DateFormatSymbols)
|
|
|
|
#define kSUPPLEMENTAL "supplementalData"
|
|
|
|
/**
|
|
* These are the tags we expect to see in normal resource bundle files associated
|
|
* with a locale and calendar
|
|
*/
|
|
static const char gErasTag[]="eras";
|
|
static const char gCyclicNameSetsTag[]="cyclicNameSets";
|
|
static const char gNameSetYearsTag[]="years";
|
|
static const char gNameSetZodiacsTag[]="zodiacs";
|
|
static const char gMonthNamesTag[]="monthNames";
|
|
static const char gMonthPatternsTag[]="monthPatterns";
|
|
static const char gDayNamesTag[]="dayNames";
|
|
static const char gNamesWideTag[]="wide";
|
|
static const char gNamesAbbrTag[]="abbreviated";
|
|
static const char gNamesShortTag[]="short";
|
|
static const char gNamesNarrowTag[]="narrow";
|
|
static const char gNamesAllTag[]="all";
|
|
static const char gNamesLeapTag[]="leap";
|
|
static const char gNamesFormatTag[]="format";
|
|
static const char gNamesStandaloneTag[]="stand-alone";
|
|
static const char gNamesNumericTag[]="numeric";
|
|
static const char gAmPmMarkersTag[]="AmPmMarkers";
|
|
static const char gAmPmMarkersNarrowTag[]="AmPmMarkersNarrow";
|
|
static const char gQuartersTag[]="quarters";
|
|
static const char gNumberElementsTag[]="NumberElements";
|
|
static const char gSymbolsTag[]="symbols";
|
|
static const char gTimeSeparatorTag[]="timeSeparator";
|
|
static const char gDayPeriodTag[]="dayPeriod";
|
|
|
|
// static const char gZoneStringsTag[]="zoneStrings";
|
|
|
|
// static const char gLocalPatternCharsTag[]="localPatternChars";
|
|
|
|
static const char gContextTransformsTag[]="contextTransforms";
|
|
|
|
static UMutex LOCK = U_MUTEX_INITIALIZER;
|
|
|
|
/**
|
|
* Jitterbug 2974: MSVC has a bug whereby new X[0] behaves badly.
|
|
* Work around this.
|
|
*/
|
|
static inline UnicodeString* newUnicodeStringArray(size_t count) {
|
|
return new UnicodeString[count ? count : 1];
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
DateFormatSymbols * U_EXPORT2
|
|
DateFormatSymbols::createForLocale(
|
|
const Locale& locale, UErrorCode &status) {
|
|
const SharedDateFormatSymbols *shared = NULL;
|
|
UnifiedCache::getByLocale(locale, shared, status);
|
|
if (U_FAILURE(status)) {
|
|
return NULL;
|
|
}
|
|
DateFormatSymbols *result = new DateFormatSymbols(shared->get());
|
|
shared->removeRef();
|
|
if (result == NULL) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
return NULL;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
DateFormatSymbols::DateFormatSymbols(const Locale& locale,
|
|
UErrorCode& status)
|
|
: UObject()
|
|
{
|
|
initializeData(locale, NULL, status);
|
|
}
|
|
|
|
DateFormatSymbols::DateFormatSymbols(UErrorCode& status)
|
|
: UObject()
|
|
{
|
|
initializeData(Locale::getDefault(), NULL, status, TRUE);
|
|
}
|
|
|
|
|
|
DateFormatSymbols::DateFormatSymbols(const Locale& locale,
|
|
const char *type,
|
|
UErrorCode& status)
|
|
: UObject()
|
|
{
|
|
initializeData(locale, type, status);
|
|
}
|
|
|
|
DateFormatSymbols::DateFormatSymbols(const char *type, UErrorCode& status)
|
|
: UObject()
|
|
{
|
|
initializeData(Locale::getDefault(), type, status, TRUE);
|
|
}
|
|
|
|
DateFormatSymbols::DateFormatSymbols(const DateFormatSymbols& other)
|
|
: UObject(other)
|
|
{
|
|
copyData(other);
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::assignArray(UnicodeString*& dstArray,
|
|
int32_t& dstCount,
|
|
const UnicodeString* srcArray,
|
|
int32_t srcCount)
|
|
{
|
|
// assignArray() is only called by copyData(), which in turn implements the
|
|
// copy constructor and the assignment operator.
|
|
// All strings in a DateFormatSymbols object are created in one of the following
|
|
// three ways that all allow to safely use UnicodeString::fastCopyFrom():
|
|
// - readonly-aliases from resource bundles
|
|
// - readonly-aliases or allocated strings from constants
|
|
// - safely cloned strings (with owned buffers) from setXYZ() functions
|
|
//
|
|
// Note that this is true for as long as DateFormatSymbols can be constructed
|
|
// only from a locale bundle or set via the cloning API,
|
|
// *and* for as long as all the strings are in *private* fields, preventing
|
|
// a subclass from creating these strings in an "unsafe" way (with respect to fastCopyFrom()).
|
|
dstCount = srcCount;
|
|
dstArray = newUnicodeStringArray(srcCount);
|
|
if(dstArray != NULL) {
|
|
int32_t i;
|
|
for(i=0; i<srcCount; ++i) {
|
|
dstArray[i].fastCopyFrom(srcArray[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Create a copy, in fZoneStrings, of the given zone strings array. The
|
|
* member variables fZoneStringsRowCount and fZoneStringsColCount should
|
|
* be set already by the caller.
|
|
*/
|
|
void
|
|
DateFormatSymbols::createZoneStrings(const UnicodeString *const * otherStrings)
|
|
{
|
|
int32_t row, col;
|
|
UBool failed = FALSE;
|
|
|
|
fZoneStrings = (UnicodeString **)uprv_malloc(fZoneStringsRowCount * sizeof(UnicodeString *));
|
|
if (fZoneStrings != NULL) {
|
|
for (row=0; row<fZoneStringsRowCount; ++row)
|
|
{
|
|
fZoneStrings[row] = newUnicodeStringArray(fZoneStringsColCount);
|
|
if (fZoneStrings[row] == NULL) {
|
|
failed = TRUE;
|
|
break;
|
|
}
|
|
for (col=0; col<fZoneStringsColCount; ++col) {
|
|
// fastCopyFrom() - see assignArray comments
|
|
fZoneStrings[row][col].fastCopyFrom(otherStrings[row][col]);
|
|
}
|
|
}
|
|
}
|
|
// If memory allocation failed, roll back and delete fZoneStrings
|
|
if (failed) {
|
|
for (int i = row; i >= 0; i--) {
|
|
delete[] fZoneStrings[i];
|
|
}
|
|
uprv_free(fZoneStrings);
|
|
fZoneStrings = NULL;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Copy all of the other's data to this.
|
|
*/
|
|
void
|
|
DateFormatSymbols::copyData(const DateFormatSymbols& other) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
U_LOCALE_BASED(locBased, *this);
|
|
locBased.setLocaleIDs(
|
|
other.getLocale(ULOC_VALID_LOCALE, status),
|
|
other.getLocale(ULOC_ACTUAL_LOCALE, status));
|
|
assignArray(fEras, fErasCount, other.fEras, other.fErasCount);
|
|
assignArray(fEraNames, fEraNamesCount, other.fEraNames, other.fEraNamesCount);
|
|
assignArray(fNarrowEras, fNarrowErasCount, other.fNarrowEras, other.fNarrowErasCount);
|
|
assignArray(fMonths, fMonthsCount, other.fMonths, other.fMonthsCount);
|
|
assignArray(fShortMonths, fShortMonthsCount, other.fShortMonths, other.fShortMonthsCount);
|
|
assignArray(fNarrowMonths, fNarrowMonthsCount, other.fNarrowMonths, other.fNarrowMonthsCount);
|
|
assignArray(fStandaloneMonths, fStandaloneMonthsCount, other.fStandaloneMonths, other.fStandaloneMonthsCount);
|
|
assignArray(fStandaloneShortMonths, fStandaloneShortMonthsCount, other.fStandaloneShortMonths, other.fStandaloneShortMonthsCount);
|
|
assignArray(fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount, other.fStandaloneNarrowMonths, other.fStandaloneNarrowMonthsCount);
|
|
assignArray(fWeekdays, fWeekdaysCount, other.fWeekdays, other.fWeekdaysCount);
|
|
assignArray(fShortWeekdays, fShortWeekdaysCount, other.fShortWeekdays, other.fShortWeekdaysCount);
|
|
assignArray(fShorterWeekdays, fShorterWeekdaysCount, other.fShorterWeekdays, other.fShorterWeekdaysCount);
|
|
assignArray(fNarrowWeekdays, fNarrowWeekdaysCount, other.fNarrowWeekdays, other.fNarrowWeekdaysCount);
|
|
assignArray(fStandaloneWeekdays, fStandaloneWeekdaysCount, other.fStandaloneWeekdays, other.fStandaloneWeekdaysCount);
|
|
assignArray(fStandaloneShortWeekdays, fStandaloneShortWeekdaysCount, other.fStandaloneShortWeekdays, other.fStandaloneShortWeekdaysCount);
|
|
assignArray(fStandaloneShorterWeekdays, fStandaloneShorterWeekdaysCount, other.fStandaloneShorterWeekdays, other.fStandaloneShorterWeekdaysCount);
|
|
assignArray(fStandaloneNarrowWeekdays, fStandaloneNarrowWeekdaysCount, other.fStandaloneNarrowWeekdays, other.fStandaloneNarrowWeekdaysCount);
|
|
assignArray(fAmPms, fAmPmsCount, other.fAmPms, other.fAmPmsCount);
|
|
assignArray(fNarrowAmPms, fNarrowAmPmsCount, other.fNarrowAmPms, other.fNarrowAmPmsCount );
|
|
fTimeSeparator.fastCopyFrom(other.fTimeSeparator); // fastCopyFrom() - see assignArray comments
|
|
assignArray(fQuarters, fQuartersCount, other.fQuarters, other.fQuartersCount);
|
|
assignArray(fShortQuarters, fShortQuartersCount, other.fShortQuarters, other.fShortQuartersCount);
|
|
assignArray(fStandaloneQuarters, fStandaloneQuartersCount, other.fStandaloneQuarters, other.fStandaloneQuartersCount);
|
|
assignArray(fStandaloneShortQuarters, fStandaloneShortQuartersCount, other.fStandaloneShortQuarters, other.fStandaloneShortQuartersCount);
|
|
assignArray(fWideDayPeriods, fWideDayPeriodsCount,
|
|
other.fWideDayPeriods, other.fWideDayPeriodsCount);
|
|
assignArray(fNarrowDayPeriods, fNarrowDayPeriodsCount,
|
|
other.fNarrowDayPeriods, other.fNarrowDayPeriodsCount);
|
|
assignArray(fAbbreviatedDayPeriods, fAbbreviatedDayPeriodsCount,
|
|
other.fAbbreviatedDayPeriods, other.fAbbreviatedDayPeriodsCount);
|
|
assignArray(fStandaloneWideDayPeriods, fStandaloneWideDayPeriodsCount,
|
|
other.fStandaloneWideDayPeriods, other.fStandaloneWideDayPeriodsCount);
|
|
assignArray(fStandaloneNarrowDayPeriods, fStandaloneNarrowDayPeriodsCount,
|
|
other.fStandaloneNarrowDayPeriods, other.fStandaloneNarrowDayPeriodsCount);
|
|
assignArray(fStandaloneAbbreviatedDayPeriods, fStandaloneAbbreviatedDayPeriodsCount,
|
|
other.fStandaloneAbbreviatedDayPeriods, other.fStandaloneAbbreviatedDayPeriodsCount);
|
|
if (other.fLeapMonthPatterns != NULL) {
|
|
assignArray(fLeapMonthPatterns, fLeapMonthPatternsCount, other.fLeapMonthPatterns, other.fLeapMonthPatternsCount);
|
|
} else {
|
|
fLeapMonthPatterns = NULL;
|
|
fLeapMonthPatternsCount = 0;
|
|
}
|
|
if (other.fShortYearNames != NULL) {
|
|
assignArray(fShortYearNames, fShortYearNamesCount, other.fShortYearNames, other.fShortYearNamesCount);
|
|
} else {
|
|
fShortYearNames = NULL;
|
|
fShortYearNamesCount = 0;
|
|
}
|
|
if (other.fShortZodiacNames != NULL) {
|
|
assignArray(fShortZodiacNames, fShortZodiacNamesCount, other.fShortZodiacNames, other.fShortZodiacNamesCount);
|
|
} else {
|
|
fShortZodiacNames = NULL;
|
|
fShortZodiacNamesCount = 0;
|
|
}
|
|
|
|
if (other.fZoneStrings != NULL) {
|
|
fZoneStringsColCount = other.fZoneStringsColCount;
|
|
fZoneStringsRowCount = other.fZoneStringsRowCount;
|
|
createZoneStrings((const UnicodeString**)other.fZoneStrings);
|
|
|
|
} else {
|
|
fZoneStrings = NULL;
|
|
fZoneStringsColCount = 0;
|
|
fZoneStringsRowCount = 0;
|
|
}
|
|
fZSFLocale = other.fZSFLocale;
|
|
// Other zone strings data is created on demand
|
|
fLocaleZoneStrings = NULL;
|
|
|
|
// fastCopyFrom() - see assignArray comments
|
|
fLocalPatternChars.fastCopyFrom(other.fLocalPatternChars);
|
|
|
|
uprv_memcpy(fCapitalization, other.fCapitalization, sizeof(fCapitalization));
|
|
}
|
|
|
|
/**
|
|
* Assignment operator.
|
|
*/
|
|
DateFormatSymbols& DateFormatSymbols::operator=(const DateFormatSymbols& other)
|
|
{
|
|
dispose();
|
|
copyData(other);
|
|
|
|
return *this;
|
|
}
|
|
|
|
DateFormatSymbols::~DateFormatSymbols()
|
|
{
|
|
dispose();
|
|
}
|
|
|
|
void DateFormatSymbols::dispose()
|
|
{
|
|
delete[] fEras;
|
|
delete[] fEraNames;
|
|
delete[] fNarrowEras;
|
|
delete[] fMonths;
|
|
delete[] fShortMonths;
|
|
delete[] fNarrowMonths;
|
|
delete[] fStandaloneMonths;
|
|
delete[] fStandaloneShortMonths;
|
|
delete[] fStandaloneNarrowMonths;
|
|
delete[] fWeekdays;
|
|
delete[] fShortWeekdays;
|
|
delete[] fShorterWeekdays;
|
|
delete[] fNarrowWeekdays;
|
|
delete[] fStandaloneWeekdays;
|
|
delete[] fStandaloneShortWeekdays;
|
|
delete[] fStandaloneShorterWeekdays;
|
|
delete[] fStandaloneNarrowWeekdays;
|
|
delete[] fAmPms;
|
|
delete[] fNarrowAmPms;
|
|
delete[] fQuarters;
|
|
delete[] fShortQuarters;
|
|
delete[] fStandaloneQuarters;
|
|
delete[] fStandaloneShortQuarters;
|
|
delete[] fLeapMonthPatterns;
|
|
delete[] fShortYearNames;
|
|
delete[] fShortZodiacNames;
|
|
delete[] fAbbreviatedDayPeriods;
|
|
delete[] fWideDayPeriods;
|
|
delete[] fNarrowDayPeriods;
|
|
delete[] fStandaloneAbbreviatedDayPeriods;
|
|
delete[] fStandaloneWideDayPeriods;
|
|
delete[] fStandaloneNarrowDayPeriods;
|
|
|
|
disposeZoneStrings();
|
|
}
|
|
|
|
void DateFormatSymbols::disposeZoneStrings()
|
|
{
|
|
if (fZoneStrings) {
|
|
for (int32_t row = 0; row < fZoneStringsRowCount; ++row) {
|
|
delete[] fZoneStrings[row];
|
|
}
|
|
uprv_free(fZoneStrings);
|
|
}
|
|
if (fLocaleZoneStrings) {
|
|
for (int32_t row = 0; row < fZoneStringsRowCount; ++row) {
|
|
delete[] fLocaleZoneStrings[row];
|
|
}
|
|
uprv_free(fLocaleZoneStrings);
|
|
}
|
|
|
|
fZoneStrings = NULL;
|
|
fLocaleZoneStrings = NULL;
|
|
fZoneStringsRowCount = 0;
|
|
fZoneStringsColCount = 0;
|
|
}
|
|
|
|
UBool
|
|
DateFormatSymbols::arrayCompare(const UnicodeString* array1,
|
|
const UnicodeString* array2,
|
|
int32_t count)
|
|
{
|
|
if (array1 == array2) return TRUE;
|
|
while (count>0)
|
|
{
|
|
--count;
|
|
if (array1[count] != array2[count]) return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
UBool
|
|
DateFormatSymbols::operator==(const DateFormatSymbols& other) const
|
|
{
|
|
// First do cheap comparisons
|
|
if (this == &other) {
|
|
return TRUE;
|
|
}
|
|
if (fErasCount == other.fErasCount &&
|
|
fEraNamesCount == other.fEraNamesCount &&
|
|
fNarrowErasCount == other.fNarrowErasCount &&
|
|
fMonthsCount == other.fMonthsCount &&
|
|
fShortMonthsCount == other.fShortMonthsCount &&
|
|
fNarrowMonthsCount == other.fNarrowMonthsCount &&
|
|
fStandaloneMonthsCount == other.fStandaloneMonthsCount &&
|
|
fStandaloneShortMonthsCount == other.fStandaloneShortMonthsCount &&
|
|
fStandaloneNarrowMonthsCount == other.fStandaloneNarrowMonthsCount &&
|
|
fWeekdaysCount == other.fWeekdaysCount &&
|
|
fShortWeekdaysCount == other.fShortWeekdaysCount &&
|
|
fShorterWeekdaysCount == other.fShorterWeekdaysCount &&
|
|
fNarrowWeekdaysCount == other.fNarrowWeekdaysCount &&
|
|
fStandaloneWeekdaysCount == other.fStandaloneWeekdaysCount &&
|
|
fStandaloneShortWeekdaysCount == other.fStandaloneShortWeekdaysCount &&
|
|
fStandaloneShorterWeekdaysCount == other.fStandaloneShorterWeekdaysCount &&
|
|
fStandaloneNarrowWeekdaysCount == other.fStandaloneNarrowWeekdaysCount &&
|
|
fAmPmsCount == other.fAmPmsCount &&
|
|
fNarrowAmPmsCount == other.fNarrowAmPmsCount &&
|
|
fQuartersCount == other.fQuartersCount &&
|
|
fShortQuartersCount == other.fShortQuartersCount &&
|
|
fStandaloneQuartersCount == other.fStandaloneQuartersCount &&
|
|
fStandaloneShortQuartersCount == other.fStandaloneShortQuartersCount &&
|
|
fLeapMonthPatternsCount == other.fLeapMonthPatternsCount &&
|
|
fShortYearNamesCount == other.fShortYearNamesCount &&
|
|
fShortZodiacNamesCount == other.fShortZodiacNamesCount &&
|
|
fAbbreviatedDayPeriodsCount == other.fAbbreviatedDayPeriodsCount &&
|
|
fWideDayPeriodsCount == other.fWideDayPeriodsCount &&
|
|
fNarrowDayPeriodsCount == other.fNarrowDayPeriodsCount &&
|
|
fStandaloneAbbreviatedDayPeriodsCount == other.fStandaloneAbbreviatedDayPeriodsCount &&
|
|
fStandaloneWideDayPeriodsCount == other.fStandaloneWideDayPeriodsCount &&
|
|
fStandaloneNarrowDayPeriodsCount == other.fStandaloneNarrowDayPeriodsCount &&
|
|
(uprv_memcmp(fCapitalization, other.fCapitalization, sizeof(fCapitalization))==0))
|
|
{
|
|
// Now compare the arrays themselves
|
|
if (arrayCompare(fEras, other.fEras, fErasCount) &&
|
|
arrayCompare(fEraNames, other.fEraNames, fEraNamesCount) &&
|
|
arrayCompare(fNarrowEras, other.fNarrowEras, fNarrowErasCount) &&
|
|
arrayCompare(fMonths, other.fMonths, fMonthsCount) &&
|
|
arrayCompare(fShortMonths, other.fShortMonths, fShortMonthsCount) &&
|
|
arrayCompare(fNarrowMonths, other.fNarrowMonths, fNarrowMonthsCount) &&
|
|
arrayCompare(fStandaloneMonths, other.fStandaloneMonths, fStandaloneMonthsCount) &&
|
|
arrayCompare(fStandaloneShortMonths, other.fStandaloneShortMonths, fStandaloneShortMonthsCount) &&
|
|
arrayCompare(fStandaloneNarrowMonths, other.fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount) &&
|
|
arrayCompare(fWeekdays, other.fWeekdays, fWeekdaysCount) &&
|
|
arrayCompare(fShortWeekdays, other.fShortWeekdays, fShortWeekdaysCount) &&
|
|
arrayCompare(fShorterWeekdays, other.fShorterWeekdays, fShorterWeekdaysCount) &&
|
|
arrayCompare(fNarrowWeekdays, other.fNarrowWeekdays, fNarrowWeekdaysCount) &&
|
|
arrayCompare(fStandaloneWeekdays, other.fStandaloneWeekdays, fStandaloneWeekdaysCount) &&
|
|
arrayCompare(fStandaloneShortWeekdays, other.fStandaloneShortWeekdays, fStandaloneShortWeekdaysCount) &&
|
|
arrayCompare(fStandaloneShorterWeekdays, other.fStandaloneShorterWeekdays, fStandaloneShorterWeekdaysCount) &&
|
|
arrayCompare(fStandaloneNarrowWeekdays, other.fStandaloneNarrowWeekdays, fStandaloneNarrowWeekdaysCount) &&
|
|
arrayCompare(fAmPms, other.fAmPms, fAmPmsCount) &&
|
|
arrayCompare(fNarrowAmPms, other.fNarrowAmPms, fNarrowAmPmsCount) &&
|
|
fTimeSeparator == other.fTimeSeparator &&
|
|
arrayCompare(fQuarters, other.fQuarters, fQuartersCount) &&
|
|
arrayCompare(fShortQuarters, other.fShortQuarters, fShortQuartersCount) &&
|
|
arrayCompare(fStandaloneQuarters, other.fStandaloneQuarters, fStandaloneQuartersCount) &&
|
|
arrayCompare(fStandaloneShortQuarters, other.fStandaloneShortQuarters, fStandaloneShortQuartersCount) &&
|
|
arrayCompare(fLeapMonthPatterns, other.fLeapMonthPatterns, fLeapMonthPatternsCount) &&
|
|
arrayCompare(fShortYearNames, other.fShortYearNames, fShortYearNamesCount) &&
|
|
arrayCompare(fShortZodiacNames, other.fShortZodiacNames, fShortZodiacNamesCount) &&
|
|
arrayCompare(fAbbreviatedDayPeriods, other.fAbbreviatedDayPeriods, fAbbreviatedDayPeriodsCount) &&
|
|
arrayCompare(fWideDayPeriods, other.fWideDayPeriods, fWideDayPeriodsCount) &&
|
|
arrayCompare(fNarrowDayPeriods, other.fNarrowDayPeriods, fNarrowDayPeriodsCount) &&
|
|
arrayCompare(fStandaloneAbbreviatedDayPeriods, other.fStandaloneAbbreviatedDayPeriods,
|
|
fStandaloneAbbreviatedDayPeriodsCount) &&
|
|
arrayCompare(fStandaloneWideDayPeriods, other.fStandaloneWideDayPeriods,
|
|
fStandaloneWideDayPeriodsCount) &&
|
|
arrayCompare(fStandaloneNarrowDayPeriods, other.fStandaloneNarrowDayPeriods,
|
|
fStandaloneWideDayPeriodsCount))
|
|
{
|
|
// Compare the contents of fZoneStrings
|
|
if (fZoneStrings == NULL && other.fZoneStrings == NULL) {
|
|
if (fZSFLocale == other.fZSFLocale) {
|
|
return TRUE;
|
|
}
|
|
} else if (fZoneStrings != NULL && other.fZoneStrings != NULL) {
|
|
if (fZoneStringsRowCount == other.fZoneStringsRowCount
|
|
&& fZoneStringsColCount == other.fZoneStringsColCount) {
|
|
UBool cmpres = TRUE;
|
|
for (int32_t i = 0; (i < fZoneStringsRowCount) && cmpres; i++) {
|
|
cmpres = arrayCompare(fZoneStrings[i], other.fZoneStrings[i], fZoneStringsColCount);
|
|
}
|
|
return cmpres;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getEras(int32_t &count) const
|
|
{
|
|
count = fErasCount;
|
|
return fEras;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getEraNames(int32_t &count) const
|
|
{
|
|
count = fEraNamesCount;
|
|
return fEraNames;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getNarrowEras(int32_t &count) const
|
|
{
|
|
count = fNarrowErasCount;
|
|
return fNarrowEras;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getMonths(int32_t &count) const
|
|
{
|
|
count = fMonthsCount;
|
|
return fMonths;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getShortMonths(int32_t &count) const
|
|
{
|
|
count = fShortMonthsCount;
|
|
return fShortMonths;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getMonths(int32_t &count, DtContextType context, DtWidthType width ) const
|
|
{
|
|
UnicodeString *returnValue = NULL;
|
|
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fMonthsCount;
|
|
returnValue = fMonths;
|
|
break;
|
|
case ABBREVIATED :
|
|
case SHORT : // no month data for this, defaults to ABBREVIATED
|
|
count = fShortMonthsCount;
|
|
returnValue = fShortMonths;
|
|
break;
|
|
case NARROW :
|
|
count = fNarrowMonthsCount;
|
|
returnValue = fNarrowMonths;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fStandaloneMonthsCount;
|
|
returnValue = fStandaloneMonths;
|
|
break;
|
|
case ABBREVIATED :
|
|
case SHORT : // no month data for this, defaults to ABBREVIATED
|
|
count = fStandaloneShortMonthsCount;
|
|
returnValue = fStandaloneShortMonths;
|
|
break;
|
|
case NARROW :
|
|
count = fStandaloneNarrowMonthsCount;
|
|
returnValue = fStandaloneNarrowMonths;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
return returnValue;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getWeekdays(int32_t &count) const
|
|
{
|
|
count = fWeekdaysCount;
|
|
return fWeekdays;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getShortWeekdays(int32_t &count) const
|
|
{
|
|
count = fShortWeekdaysCount;
|
|
return fShortWeekdays;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getWeekdays(int32_t &count, DtContextType context, DtWidthType width) const
|
|
{
|
|
UnicodeString *returnValue = NULL;
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fWeekdaysCount;
|
|
returnValue = fWeekdays;
|
|
break;
|
|
case ABBREVIATED :
|
|
count = fShortWeekdaysCount;
|
|
returnValue = fShortWeekdays;
|
|
break;
|
|
case SHORT :
|
|
count = fShorterWeekdaysCount;
|
|
returnValue = fShorterWeekdays;
|
|
break;
|
|
case NARROW :
|
|
count = fNarrowWeekdaysCount;
|
|
returnValue = fNarrowWeekdays;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fStandaloneWeekdaysCount;
|
|
returnValue = fStandaloneWeekdays;
|
|
break;
|
|
case ABBREVIATED :
|
|
count = fStandaloneShortWeekdaysCount;
|
|
returnValue = fStandaloneShortWeekdays;
|
|
break;
|
|
case SHORT :
|
|
count = fStandaloneShorterWeekdaysCount;
|
|
returnValue = fStandaloneShorterWeekdays;
|
|
break;
|
|
case NARROW :
|
|
count = fStandaloneNarrowWeekdaysCount;
|
|
returnValue = fStandaloneNarrowWeekdays;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
return returnValue;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getQuarters(int32_t &count, DtContextType context, DtWidthType width ) const
|
|
{
|
|
UnicodeString *returnValue = NULL;
|
|
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fQuartersCount;
|
|
returnValue = fQuarters;
|
|
break;
|
|
case ABBREVIATED :
|
|
case SHORT : // no quarter data for this, defaults to ABBREVIATED
|
|
count = fShortQuartersCount;
|
|
returnValue = fShortQuarters;
|
|
break;
|
|
case NARROW :
|
|
count = 0;
|
|
returnValue = NULL;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch(width) {
|
|
case WIDE :
|
|
count = fStandaloneQuartersCount;
|
|
returnValue = fStandaloneQuarters;
|
|
break;
|
|
case ABBREVIATED :
|
|
case SHORT : // no quarter data for this, defaults to ABBREVIATED
|
|
count = fStandaloneShortQuartersCount;
|
|
returnValue = fStandaloneShortQuarters;
|
|
break;
|
|
case NARROW :
|
|
count = 0;
|
|
returnValue = NULL;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
return returnValue;
|
|
}
|
|
|
|
UnicodeString&
|
|
DateFormatSymbols::getTimeSeparatorString(UnicodeString& result) const
|
|
{
|
|
// fastCopyFrom() - see assignArray comments
|
|
return result.fastCopyFrom(fTimeSeparator);
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getAmPmStrings(int32_t &count) const
|
|
{
|
|
count = fAmPmsCount;
|
|
return fAmPms;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getLeapMonthPatterns(int32_t &count) const
|
|
{
|
|
count = fLeapMonthPatternsCount;
|
|
return fLeapMonthPatterns;
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getYearNames(int32_t& count,
|
|
DtContextType /*ignored*/, DtWidthType /*ignored*/) const
|
|
{
|
|
count = fShortYearNamesCount;
|
|
return fShortYearNames;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setYearNames(const UnicodeString* yearNames, int32_t count,
|
|
DtContextType context, DtWidthType width)
|
|
{
|
|
if (context == FORMAT && width == ABBREVIATED) {
|
|
if (fShortYearNames) {
|
|
delete[] fShortYearNames;
|
|
}
|
|
fShortYearNames = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(yearNames, fShortYearNames, count);
|
|
fShortYearNamesCount = count;
|
|
}
|
|
}
|
|
|
|
const UnicodeString*
|
|
DateFormatSymbols::getZodiacNames(int32_t& count,
|
|
DtContextType /*ignored*/, DtWidthType /*ignored*/) const
|
|
{
|
|
count = fShortZodiacNamesCount;
|
|
return fShortZodiacNames;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setZodiacNames(const UnicodeString* zodiacNames, int32_t count,
|
|
DtContextType context, DtWidthType width)
|
|
{
|
|
if (context == FORMAT && width == ABBREVIATED) {
|
|
if (fShortZodiacNames) {
|
|
delete[] fShortZodiacNames;
|
|
}
|
|
fShortZodiacNames = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(zodiacNames, fShortZodiacNames, count);
|
|
fShortZodiacNamesCount = count;
|
|
}
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
void
|
|
DateFormatSymbols::setEras(const UnicodeString* erasArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fEras)
|
|
delete[] fEras;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fEras = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(erasArray,fEras, count);
|
|
fErasCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setEraNames(const UnicodeString* eraNamesArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fEraNames)
|
|
delete[] fEraNames;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fEraNames = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(eraNamesArray,fEraNames, count);
|
|
fEraNamesCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setNarrowEras(const UnicodeString* narrowErasArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fNarrowEras)
|
|
delete[] fNarrowEras;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fNarrowEras = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(narrowErasArray,fNarrowEras, count);
|
|
fNarrowErasCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setMonths(const UnicodeString* monthsArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fMonths)
|
|
delete[] fMonths;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fMonths,count);
|
|
fMonthsCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setShortMonths(const UnicodeString* shortMonthsArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fShortMonths)
|
|
delete[] fShortMonths;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fShortMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(shortMonthsArray,fShortMonths, count);
|
|
fShortMonthsCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setMonths(const UnicodeString* monthsArray, int32_t count, DtContextType context, DtWidthType width)
|
|
{
|
|
// delete the old list if we own it
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fMonths)
|
|
delete[] fMonths;
|
|
fMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fMonths,count);
|
|
fMonthsCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fShortMonths)
|
|
delete[] fShortMonths;
|
|
fShortMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fShortMonths,count);
|
|
fShortMonthsCount = count;
|
|
break;
|
|
case NARROW :
|
|
if (fNarrowMonths)
|
|
delete[] fNarrowMonths;
|
|
fNarrowMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fNarrowMonths,count);
|
|
fNarrowMonthsCount = count;
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fStandaloneMonths)
|
|
delete[] fStandaloneMonths;
|
|
fStandaloneMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fStandaloneMonths,count);
|
|
fStandaloneMonthsCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fStandaloneShortMonths)
|
|
delete[] fStandaloneShortMonths;
|
|
fStandaloneShortMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fStandaloneShortMonths,count);
|
|
fStandaloneShortMonthsCount = count;
|
|
break;
|
|
case NARROW :
|
|
if (fStandaloneNarrowMonths)
|
|
delete[] fStandaloneNarrowMonths;
|
|
fStandaloneNarrowMonths = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( monthsArray,fStandaloneNarrowMonths,count);
|
|
fStandaloneNarrowMonthsCount = count;
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
}
|
|
|
|
void DateFormatSymbols::setWeekdays(const UnicodeString* weekdaysArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fWeekdays)
|
|
delete[] fWeekdays;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray,fWeekdays,count);
|
|
fWeekdaysCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setShortWeekdays(const UnicodeString* shortWeekdaysArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fShortWeekdays)
|
|
delete[] fShortWeekdays;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fShortWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(shortWeekdaysArray, fShortWeekdays, count);
|
|
fShortWeekdaysCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setWeekdays(const UnicodeString* weekdaysArray, int32_t count, DtContextType context, DtWidthType width)
|
|
{
|
|
// delete the old list if we own it
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fWeekdays)
|
|
delete[] fWeekdays;
|
|
fWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fWeekdays, count);
|
|
fWeekdaysCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fShortWeekdays)
|
|
delete[] fShortWeekdays;
|
|
fShortWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fShortWeekdays, count);
|
|
fShortWeekdaysCount = count;
|
|
break;
|
|
case SHORT :
|
|
if (fShorterWeekdays)
|
|
delete[] fShorterWeekdays;
|
|
fShorterWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fShorterWeekdays, count);
|
|
fShorterWeekdaysCount = count;
|
|
break;
|
|
case NARROW :
|
|
if (fNarrowWeekdays)
|
|
delete[] fNarrowWeekdays;
|
|
fNarrowWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fNarrowWeekdays, count);
|
|
fNarrowWeekdaysCount = count;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fStandaloneWeekdays)
|
|
delete[] fStandaloneWeekdays;
|
|
fStandaloneWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fStandaloneWeekdays, count);
|
|
fStandaloneWeekdaysCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fStandaloneShortWeekdays)
|
|
delete[] fStandaloneShortWeekdays;
|
|
fStandaloneShortWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fStandaloneShortWeekdays, count);
|
|
fStandaloneShortWeekdaysCount = count;
|
|
break;
|
|
case SHORT :
|
|
if (fStandaloneShorterWeekdays)
|
|
delete[] fStandaloneShorterWeekdays;
|
|
fStandaloneShorterWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fStandaloneShorterWeekdays, count);
|
|
fStandaloneShorterWeekdaysCount = count;
|
|
break;
|
|
case NARROW :
|
|
if (fStandaloneNarrowWeekdays)
|
|
delete[] fStandaloneNarrowWeekdays;
|
|
fStandaloneNarrowWeekdays = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(weekdaysArray, fStandaloneNarrowWeekdays, count);
|
|
fStandaloneNarrowWeekdaysCount = count;
|
|
break;
|
|
case DT_WIDTH_COUNT :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setQuarters(const UnicodeString* quartersArray, int32_t count, DtContextType context, DtWidthType width)
|
|
{
|
|
// delete the old list if we own it
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
|
|
switch (context) {
|
|
case FORMAT :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fQuarters)
|
|
delete[] fQuarters;
|
|
fQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fQuarters,count);
|
|
fQuartersCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fShortQuarters)
|
|
delete[] fShortQuarters;
|
|
fShortQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fShortQuarters,count);
|
|
fShortQuartersCount = count;
|
|
break;
|
|
case NARROW :
|
|
/*
|
|
if (fNarrowQuarters)
|
|
delete[] fNarrowQuarters;
|
|
fNarrowQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fNarrowQuarters,count);
|
|
fNarrowQuartersCount = count;
|
|
*/
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
break;
|
|
case STANDALONE :
|
|
switch (width) {
|
|
case WIDE :
|
|
if (fStandaloneQuarters)
|
|
delete[] fStandaloneQuarters;
|
|
fStandaloneQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fStandaloneQuarters,count);
|
|
fStandaloneQuartersCount = count;
|
|
break;
|
|
case ABBREVIATED :
|
|
if (fStandaloneShortQuarters)
|
|
delete[] fStandaloneShortQuarters;
|
|
fStandaloneShortQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fStandaloneShortQuarters,count);
|
|
fStandaloneShortQuartersCount = count;
|
|
break;
|
|
case NARROW :
|
|
/*
|
|
if (fStandaloneNarrowQuarters)
|
|
delete[] fStandaloneNarrowQuarters;
|
|
fStandaloneNarrowQuarters = newUnicodeStringArray(count);
|
|
uprv_arrayCopy( quartersArray,fStandaloneNarrowQuarters,count);
|
|
fStandaloneNarrowQuartersCount = count;
|
|
*/
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
break;
|
|
case DT_CONTEXT_COUNT :
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setAmPmStrings(const UnicodeString* amPmsArray, int32_t count)
|
|
{
|
|
// delete the old list if we own it
|
|
if (fAmPms) delete[] fAmPms;
|
|
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fAmPms = newUnicodeStringArray(count);
|
|
uprv_arrayCopy(amPmsArray,fAmPms,count);
|
|
fAmPmsCount = count;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setTimeSeparatorString(const UnicodeString& newTimeSeparator)
|
|
{
|
|
fTimeSeparator = newTimeSeparator;
|
|
}
|
|
|
|
const UnicodeString**
|
|
DateFormatSymbols::getZoneStrings(int32_t& rowCount, int32_t& columnCount) const
|
|
{
|
|
const UnicodeString **result = NULL;
|
|
|
|
umtx_lock(&LOCK);
|
|
if (fZoneStrings == NULL) {
|
|
if (fLocaleZoneStrings == NULL) {
|
|
((DateFormatSymbols*)this)->initZoneStringsArray();
|
|
}
|
|
result = (const UnicodeString**)fLocaleZoneStrings;
|
|
} else {
|
|
result = (const UnicodeString**)fZoneStrings;
|
|
}
|
|
rowCount = fZoneStringsRowCount;
|
|
columnCount = fZoneStringsColCount;
|
|
umtx_unlock(&LOCK);
|
|
|
|
return result;
|
|
}
|
|
|
|
// For now, we include all zones
|
|
#define ZONE_SET UCAL_ZONE_TYPE_ANY
|
|
|
|
// This code must be called within a synchronized block
|
|
void
|
|
DateFormatSymbols::initZoneStringsArray(void) {
|
|
if (fZoneStrings != NULL || fLocaleZoneStrings != NULL) {
|
|
return;
|
|
}
|
|
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
|
|
StringEnumeration *tzids = NULL;
|
|
UnicodeString ** zarray = NULL;
|
|
TimeZoneNames *tzNames = NULL;
|
|
int32_t rows = 0;
|
|
|
|
static const UTimeZoneNameType TYPES[] = {
|
|
UTZNM_LONG_STANDARD, UTZNM_SHORT_STANDARD,
|
|
UTZNM_LONG_DAYLIGHT, UTZNM_SHORT_DAYLIGHT
|
|
};
|
|
static const int32_t NUM_TYPES = 4;
|
|
|
|
do { // dummy do-while
|
|
|
|
tzids = TimeZone::createTimeZoneIDEnumeration(ZONE_SET, NULL, NULL, status);
|
|
rows = tzids->count(status);
|
|
if (U_FAILURE(status)) {
|
|
break;
|
|
}
|
|
|
|
// Allocate array
|
|
int32_t size = rows * sizeof(UnicodeString*);
|
|
zarray = (UnicodeString**)uprv_malloc(size);
|
|
if (zarray == NULL) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
break;
|
|
}
|
|
uprv_memset(zarray, 0, size);
|
|
|
|
tzNames = TimeZoneNames::createInstance(fZSFLocale, status);
|
|
tzNames->loadAllDisplayNames(status);
|
|
if (U_FAILURE(status)) { break; }
|
|
|
|
const UnicodeString *tzid;
|
|
int32_t i = 0;
|
|
UDate now = Calendar::getNow();
|
|
UnicodeString tzDispName;
|
|
|
|
while ((tzid = tzids->snext(status))) {
|
|
if (U_FAILURE(status)) {
|
|
break;
|
|
}
|
|
|
|
zarray[i] = new UnicodeString[5];
|
|
if (zarray[i] == NULL) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
break;
|
|
}
|
|
|
|
zarray[i][0].setTo(*tzid);
|
|
tzNames->getDisplayNames(*tzid, TYPES, NUM_TYPES, now, zarray[i]+1);
|
|
i++;
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
if (U_FAILURE(status)) {
|
|
if (zarray) {
|
|
for (int32_t i = 0; i < rows; i++) {
|
|
if (zarray[i]) {
|
|
delete[] zarray[i];
|
|
}
|
|
}
|
|
uprv_free(zarray);
|
|
}
|
|
}
|
|
|
|
if (tzNames) {
|
|
delete tzNames;
|
|
}
|
|
if (tzids) {
|
|
delete tzids;
|
|
}
|
|
|
|
fLocaleZoneStrings = zarray;
|
|
fZoneStringsRowCount = rows;
|
|
fZoneStringsColCount = 1 + NUM_TYPES;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::setZoneStrings(const UnicodeString* const *strings, int32_t rowCount, int32_t columnCount)
|
|
{
|
|
// since deleting a 2-d array is a pain in the butt, we offload that task to
|
|
// a separate function
|
|
disposeZoneStrings();
|
|
// we always own the new list, which we create here (we duplicate rather
|
|
// than adopting the list passed in)
|
|
fZoneStringsRowCount = rowCount;
|
|
fZoneStringsColCount = columnCount;
|
|
createZoneStrings((const UnicodeString**)strings);
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
const UChar * U_EXPORT2
|
|
DateFormatSymbols::getPatternUChars(void)
|
|
{
|
|
return gPatternChars;
|
|
}
|
|
|
|
UDateFormatField U_EXPORT2
|
|
DateFormatSymbols::getPatternCharIndex(UChar c) {
|
|
const UChar *p = u_strchr(gPatternChars, c);
|
|
if (p == NULL) {
|
|
return UDAT_FIELD_COUNT;
|
|
} else {
|
|
return static_cast<UDateFormatField>(p - gPatternChars);
|
|
}
|
|
}
|
|
|
|
static const uint64_t kNumericFieldsAlways =
|
|
((uint64_t)1 << UDAT_YEAR_FIELD) | // y
|
|
((uint64_t)1 << UDAT_DATE_FIELD) | // d
|
|
((uint64_t)1 << UDAT_HOUR_OF_DAY1_FIELD) | // k
|
|
((uint64_t)1 << UDAT_HOUR_OF_DAY0_FIELD) | // H
|
|
((uint64_t)1 << UDAT_MINUTE_FIELD) | // m
|
|
((uint64_t)1 << UDAT_SECOND_FIELD) | // s
|
|
((uint64_t)1 << UDAT_FRACTIONAL_SECOND_FIELD) | // S
|
|
((uint64_t)1 << UDAT_DAY_OF_YEAR_FIELD) | // D
|
|
((uint64_t)1 << UDAT_DAY_OF_WEEK_IN_MONTH_FIELD) | // F
|
|
((uint64_t)1 << UDAT_WEEK_OF_YEAR_FIELD) | // w
|
|
((uint64_t)1 << UDAT_WEEK_OF_MONTH_FIELD) | // W
|
|
((uint64_t)1 << UDAT_HOUR1_FIELD) | // h
|
|
((uint64_t)1 << UDAT_HOUR0_FIELD) | // K
|
|
((uint64_t)1 << UDAT_YEAR_WOY_FIELD) | // Y
|
|
((uint64_t)1 << UDAT_EXTENDED_YEAR_FIELD) | // u
|
|
((uint64_t)1 << UDAT_JULIAN_DAY_FIELD) | // g
|
|
((uint64_t)1 << UDAT_MILLISECONDS_IN_DAY_FIELD) | // A
|
|
((uint64_t)1 << UDAT_RELATED_YEAR_FIELD); // r
|
|
|
|
static const uint64_t kNumericFieldsForCount12 =
|
|
((uint64_t)1 << UDAT_MONTH_FIELD) | // M or MM
|
|
((uint64_t)1 << UDAT_DOW_LOCAL_FIELD) | // e or ee
|
|
((uint64_t)1 << UDAT_STANDALONE_DAY_FIELD) | // c or cc
|
|
((uint64_t)1 << UDAT_STANDALONE_MONTH_FIELD) | // L or LL
|
|
((uint64_t)1 << UDAT_QUARTER_FIELD) | // Q or QQ
|
|
((uint64_t)1 << UDAT_STANDALONE_QUARTER_FIELD); // q or qq
|
|
|
|
UBool U_EXPORT2
|
|
DateFormatSymbols::isNumericField(UDateFormatField f, int32_t count) {
|
|
if (f == UDAT_FIELD_COUNT) {
|
|
return FALSE;
|
|
}
|
|
uint64_t flag = ((uint64_t)1 << f);
|
|
return ((kNumericFieldsAlways & flag) != 0 || ((kNumericFieldsForCount12 & flag) != 0 && count < 3));
|
|
}
|
|
|
|
UBool U_EXPORT2
|
|
DateFormatSymbols::isNumericPatternChar(UChar c, int32_t count) {
|
|
return isNumericField(getPatternCharIndex(c), count);
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
UnicodeString&
|
|
DateFormatSymbols::getLocalPatternChars(UnicodeString& result) const
|
|
{
|
|
// fastCopyFrom() - see assignArray comments
|
|
return result.fastCopyFrom(fLocalPatternChars);
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
void
|
|
DateFormatSymbols::setLocalPatternChars(const UnicodeString& newLocalPatternChars)
|
|
{
|
|
fLocalPatternChars = newLocalPatternChars;
|
|
}
|
|
|
|
//------------------------------------------------------
|
|
|
|
static void
|
|
initField(UnicodeString **field, int32_t& length, const UResourceBundle *data, UErrorCode &status) {
|
|
if (U_SUCCESS(status)) {
|
|
int32_t strLen = 0;
|
|
length = ures_getSize(data);
|
|
*field = newUnicodeStringArray(length);
|
|
if (*field) {
|
|
for(int32_t i = 0; i<length; i++) {
|
|
const UChar *resStr = ures_getStringByIndex(data, i, &strLen, &status);
|
|
// setTo() - see assignArray comments
|
|
(*(field)+i)->setTo(TRUE, resStr, strLen);
|
|
}
|
|
}
|
|
else {
|
|
length = 0;
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
initField(UnicodeString **field, int32_t& length, const UChar *data, LastResortSize numStr, LastResortSize strLen, UErrorCode &status) {
|
|
if (U_SUCCESS(status)) {
|
|
length = numStr;
|
|
*field = newUnicodeStringArray((size_t)numStr);
|
|
if (*field) {
|
|
for(int32_t i = 0; i<length; i++) {
|
|
// readonly aliases - all "data" strings are constant
|
|
// -1 as length for variable-length strings (gLastResortDayNames[0] is empty)
|
|
(*(field)+i)->setTo(TRUE, data+(i*((int32_t)strLen)), -1);
|
|
}
|
|
}
|
|
else {
|
|
length = 0;
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
initLeapMonthPattern(UnicodeString *field, int32_t index, const UResourceBundle *data, UErrorCode &status) {
|
|
field[index].remove();
|
|
if (U_SUCCESS(status)) {
|
|
int32_t strLen = 0;
|
|
const UChar *resStr = ures_getStringByKey(data, gNamesLeapTag, &strLen, &status);
|
|
if (U_SUCCESS(status)) {
|
|
field[index].setTo(TRUE, resStr, strLen);
|
|
}
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
|
|
typedef struct {
|
|
const char * usageTypeName;
|
|
DateFormatSymbols::ECapitalizationContextUsageType usageTypeEnumValue;
|
|
} ContextUsageTypeNameToEnumValue;
|
|
|
|
static const ContextUsageTypeNameToEnumValue contextUsageTypeMap[] = {
|
|
// Entries must be sorted by usageTypeName; entry with NULL name terminates list.
|
|
{ "day-format-except-narrow", DateFormatSymbols::kCapContextUsageDayFormat },
|
|
{ "day-narrow", DateFormatSymbols::kCapContextUsageDayNarrow },
|
|
{ "day-standalone-except-narrow", DateFormatSymbols::kCapContextUsageDayStandalone },
|
|
{ "era-abbr", DateFormatSymbols::kCapContextUsageEraAbbrev },
|
|
{ "era-name", DateFormatSymbols::kCapContextUsageEraWide },
|
|
{ "era-narrow", DateFormatSymbols::kCapContextUsageEraNarrow },
|
|
{ "metazone-long", DateFormatSymbols::kCapContextUsageMetazoneLong },
|
|
{ "metazone-short", DateFormatSymbols::kCapContextUsageMetazoneShort },
|
|
{ "month-format-except-narrow", DateFormatSymbols::kCapContextUsageMonthFormat },
|
|
{ "month-narrow", DateFormatSymbols::kCapContextUsageMonthNarrow },
|
|
{ "month-standalone-except-narrow", DateFormatSymbols::kCapContextUsageMonthStandalone },
|
|
{ "zone-long", DateFormatSymbols::kCapContextUsageZoneLong },
|
|
{ "zone-short", DateFormatSymbols::kCapContextUsageZoneShort },
|
|
{ NULL, (DateFormatSymbols::ECapitalizationContextUsageType)0 },
|
|
};
|
|
|
|
// Resource keys to look up localized strings for day periods.
|
|
// The first one must be midnight and the second must be noon, so that their indices coincide
|
|
// with the am/pm field. Formatting and parsing code for day periods relies on this coincidence.
|
|
static const char *dayPeriodKeys[] = {"midnight", "noon",
|
|
"morning1", "afternoon1", "evening1", "night1",
|
|
"morning2", "afternoon2", "evening2", "night2"};
|
|
|
|
UnicodeString* loadDayPeriodStrings(CalendarData &calData, const char *tag, UBool standalone,
|
|
int32_t &stringCount, UErrorCode &status) {
|
|
if (U_FAILURE(status)) {
|
|
return NULL;
|
|
}
|
|
|
|
UResourceBundle *dayPeriodData;
|
|
|
|
if (standalone) {
|
|
dayPeriodData = calData.getByKey3(gDayPeriodTag, gNamesStandaloneTag, tag, status);
|
|
} else {
|
|
dayPeriodData = calData.getByKey2(gDayPeriodTag, tag, status);
|
|
}
|
|
|
|
stringCount = UPRV_LENGTHOF(dayPeriodKeys);
|
|
UnicodeString *strings = new UnicodeString[stringCount];
|
|
for (int32_t i = 0; i < stringCount; ++i) {
|
|
//TODO: Check if there are fallbacks/aliases defined in the data; e.g., if there
|
|
//is no wide string, then use the narrow one?
|
|
strings[i].fastCopyFrom(ures_getUnicodeStringByKey(dayPeriodData, dayPeriodKeys[i], &status));
|
|
if (U_FAILURE(status)) {
|
|
// string[i] will be bogus if ures_getUnicodeString() returns with an error,
|
|
// which is just the behavior we want. Simply reset the error code.
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
}
|
|
return strings;
|
|
}
|
|
|
|
void
|
|
DateFormatSymbols::initializeData(const Locale& locale, const char *type, UErrorCode& status, UBool useLastResortData)
|
|
{
|
|
int32_t i;
|
|
int32_t len = 0;
|
|
const UChar *resStr;
|
|
/* In case something goes wrong, initialize all of the data to NULL. */
|
|
fEras = NULL;
|
|
fErasCount = 0;
|
|
fEraNames = NULL;
|
|
fEraNamesCount = 0;
|
|
fNarrowEras = NULL;
|
|
fNarrowErasCount = 0;
|
|
fMonths = NULL;
|
|
fMonthsCount=0;
|
|
fShortMonths = NULL;
|
|
fShortMonthsCount=0;
|
|
fNarrowMonths = NULL;
|
|
fNarrowMonthsCount=0;
|
|
fStandaloneMonths = NULL;
|
|
fStandaloneMonthsCount=0;
|
|
fStandaloneShortMonths = NULL;
|
|
fStandaloneShortMonthsCount=0;
|
|
fStandaloneNarrowMonths = NULL;
|
|
fStandaloneNarrowMonthsCount=0;
|
|
fWeekdays = NULL;
|
|
fWeekdaysCount=0;
|
|
fShortWeekdays = NULL;
|
|
fShortWeekdaysCount=0;
|
|
fShorterWeekdays = NULL;
|
|
fShorterWeekdaysCount=0;
|
|
fNarrowWeekdays = NULL;
|
|
fNarrowWeekdaysCount=0;
|
|
fStandaloneWeekdays = NULL;
|
|
fStandaloneWeekdaysCount=0;
|
|
fStandaloneShortWeekdays = NULL;
|
|
fStandaloneShortWeekdaysCount=0;
|
|
fStandaloneShorterWeekdays = NULL;
|
|
fStandaloneShorterWeekdaysCount=0;
|
|
fStandaloneNarrowWeekdays = NULL;
|
|
fStandaloneNarrowWeekdaysCount=0;
|
|
fAmPms = NULL;
|
|
fAmPmsCount=0;
|
|
fNarrowAmPms = NULL;
|
|
fNarrowAmPmsCount=0;
|
|
fTimeSeparator.setToBogus();
|
|
fQuarters = NULL;
|
|
fQuartersCount = 0;
|
|
fShortQuarters = NULL;
|
|
fShortQuartersCount = 0;
|
|
fStandaloneQuarters = NULL;
|
|
fStandaloneQuartersCount = 0;
|
|
fStandaloneShortQuarters = NULL;
|
|
fStandaloneShortQuartersCount = 0;
|
|
fLeapMonthPatterns = NULL;
|
|
fLeapMonthPatternsCount = 0;
|
|
fShortYearNames = NULL;
|
|
fShortYearNamesCount = 0;
|
|
fShortZodiacNames = NULL;
|
|
fShortZodiacNamesCount = 0;
|
|
fZoneStringsRowCount = 0;
|
|
fZoneStringsColCount = 0;
|
|
fZoneStrings = NULL;
|
|
fLocaleZoneStrings = NULL;
|
|
fAbbreviatedDayPeriods = NULL;
|
|
fAbbreviatedDayPeriodsCount = 0;
|
|
fWideDayPeriods = NULL;
|
|
fWideDayPeriodsCount = 0;
|
|
fNarrowDayPeriods = NULL;
|
|
fNarrowDayPeriodsCount = 0;
|
|
fStandaloneAbbreviatedDayPeriods = NULL;
|
|
fStandaloneAbbreviatedDayPeriodsCount = 0;
|
|
fStandaloneWideDayPeriods = NULL;
|
|
fStandaloneWideDayPeriodsCount = 0;
|
|
fStandaloneNarrowDayPeriods = NULL;
|
|
fStandaloneNarrowDayPeriodsCount = 0;
|
|
uprv_memset(fCapitalization, 0, sizeof(fCapitalization));
|
|
|
|
// We need to preserve the requested locale for
|
|
// lazy ZoneStringFormat instantiation. ZoneStringFormat
|
|
// is region sensitive, thus, bundle locale bundle's locale
|
|
// is not sufficient.
|
|
fZSFLocale = locale;
|
|
|
|
if (U_FAILURE(status)) return;
|
|
|
|
/**
|
|
* Retrieve the string arrays we need from the resource bundle file.
|
|
* We cast away const here, but that's okay; we won't delete any of
|
|
* these.
|
|
*/
|
|
CalendarData calData(locale, type, status);
|
|
|
|
// load the first data item
|
|
UResourceBundle *erasMain = calData.getByKey(gErasTag, status);
|
|
UResourceBundle *eras = ures_getByKeyWithFallback(erasMain, gNamesAbbrTag, NULL, &status);
|
|
UErrorCode oldStatus = status;
|
|
UResourceBundle *eraNames = ures_getByKeyWithFallback(erasMain, gNamesWideTag, NULL, &status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { // Workaround because eras/wide was omitted from CLDR 1.3
|
|
status = oldStatus;
|
|
eraNames = ures_getByKeyWithFallback(erasMain, gNamesAbbrTag, NULL, &status);
|
|
}
|
|
// current ICU4J falls back to abbreviated if narrow eras are missing, so we will too
|
|
oldStatus = status;
|
|
UResourceBundle *narrowEras = ures_getByKeyWithFallback(erasMain, gNamesNarrowTag, NULL, &status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = oldStatus;
|
|
narrowEras = ures_getByKeyWithFallback(erasMain, gNamesAbbrTag, NULL, &status);
|
|
}
|
|
|
|
UErrorCode tempStatus = U_ZERO_ERROR;
|
|
UResourceBundle *monthPatterns = calData.getByKey(gMonthPatternsTag, tempStatus);
|
|
if (U_SUCCESS(tempStatus) && monthPatterns != NULL) {
|
|
fLeapMonthPatterns = newUnicodeStringArray(kMonthPatternsCount);
|
|
if (fLeapMonthPatterns) {
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternFormatWide, calData.getByKey2(gMonthPatternsTag, gNamesWideTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternFormatAbbrev, calData.getByKey2(gMonthPatternsTag, gNamesAbbrTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternFormatNarrow, calData.getByKey2(gMonthPatternsTag, gNamesNarrowTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternStandaloneWide, calData.getByKey3(gMonthPatternsTag, gNamesStandaloneTag, gNamesWideTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternStandaloneAbbrev, calData.getByKey3(gMonthPatternsTag, gNamesStandaloneTag, gNamesAbbrTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternStandaloneNarrow, calData.getByKey3(gMonthPatternsTag, gNamesStandaloneTag, gNamesNarrowTag, tempStatus), tempStatus);
|
|
initLeapMonthPattern(fLeapMonthPatterns, kLeapMonthPatternNumeric, calData.getByKey3(gMonthPatternsTag, gNamesNumericTag, gNamesAllTag, tempStatus), tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
// Hack to fix bad C inheritance for dangi monthPatterns (OK in J); this should be handled by aliases in root, but isn't.
|
|
// The ordering of the following statements is important.
|
|
if (fLeapMonthPatterns[kLeapMonthPatternFormatAbbrev].isEmpty()) {
|
|
fLeapMonthPatterns[kLeapMonthPatternFormatAbbrev].setTo(fLeapMonthPatterns[kLeapMonthPatternFormatWide]);
|
|
};
|
|
if (fLeapMonthPatterns[kLeapMonthPatternFormatNarrow].isEmpty()) {
|
|
fLeapMonthPatterns[kLeapMonthPatternFormatNarrow].setTo(fLeapMonthPatterns[kLeapMonthPatternStandaloneNarrow]);
|
|
};
|
|
if (fLeapMonthPatterns[kLeapMonthPatternStandaloneWide].isEmpty()) {
|
|
fLeapMonthPatterns[kLeapMonthPatternStandaloneWide].setTo(fLeapMonthPatterns[kLeapMonthPatternFormatWide]);
|
|
};
|
|
if (fLeapMonthPatterns[kLeapMonthPatternStandaloneAbbrev].isEmpty()) {
|
|
fLeapMonthPatterns[kLeapMonthPatternStandaloneAbbrev].setTo(fLeapMonthPatterns[kLeapMonthPatternFormatAbbrev]);
|
|
};
|
|
// end of hack
|
|
fLeapMonthPatternsCount = kMonthPatternsCount;
|
|
} else {
|
|
delete[] fLeapMonthPatterns;
|
|
fLeapMonthPatterns = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
tempStatus = U_ZERO_ERROR;
|
|
UResourceBundle *cyclicNameSets= calData.getByKey(gCyclicNameSetsTag, tempStatus);
|
|
if (U_SUCCESS(tempStatus) && cyclicNameSets != NULL) {
|
|
UResourceBundle *nameSetYears = ures_getByKeyWithFallback(cyclicNameSets, gNameSetYearsTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *nameSetYearsFmt = ures_getByKeyWithFallback(nameSetYears, gNamesFormatTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *nameSetYearsFmtAbbrev = ures_getByKeyWithFallback(nameSetYearsFmt, gNamesAbbrTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
initField(&fShortYearNames, fShortYearNamesCount, nameSetYearsFmtAbbrev, tempStatus);
|
|
ures_close(nameSetYearsFmtAbbrev);
|
|
}
|
|
ures_close(nameSetYearsFmt);
|
|
}
|
|
ures_close(nameSetYears);
|
|
}
|
|
UResourceBundle *nameSetZodiacs = ures_getByKeyWithFallback(cyclicNameSets, gNameSetZodiacsTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *nameSetZodiacsFmt = ures_getByKeyWithFallback(nameSetZodiacs, gNamesFormatTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *nameSetZodiacsFmtAbbrev = ures_getByKeyWithFallback(nameSetZodiacsFmt, gNamesAbbrTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
initField(&fShortZodiacNames, fShortZodiacNamesCount, nameSetZodiacsFmtAbbrev, tempStatus);
|
|
ures_close(nameSetZodiacsFmtAbbrev);
|
|
}
|
|
ures_close(nameSetZodiacsFmt);
|
|
}
|
|
ures_close(nameSetZodiacs);
|
|
}
|
|
}
|
|
|
|
tempStatus = U_ZERO_ERROR;
|
|
UResourceBundle *localeBundle = ures_open(NULL, locale.getName(), &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *contextTransforms = ures_getByKeyWithFallback(localeBundle, gContextTransformsTag, NULL, &tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
UResourceBundle *contextTransformUsage;
|
|
while ( (contextTransformUsage = ures_getNextResource(contextTransforms, NULL, &tempStatus)) != NULL ) {
|
|
const int32_t * intVector = ures_getIntVector(contextTransformUsage, &len, &status);
|
|
if (U_SUCCESS(tempStatus) && intVector != NULL && len >= 2) {
|
|
const char* usageType = ures_getKey(contextTransformUsage);
|
|
if (usageType != NULL) {
|
|
const ContextUsageTypeNameToEnumValue * typeMapPtr = contextUsageTypeMap;
|
|
int32_t compResult = 0;
|
|
// linear search; list is short and we cannot be sure that bsearch is available
|
|
while ( typeMapPtr->usageTypeName != NULL && (compResult = uprv_strcmp(usageType, typeMapPtr->usageTypeName)) > 0 ) {
|
|
++typeMapPtr;
|
|
}
|
|
if (typeMapPtr->usageTypeName != NULL && compResult == 0) {
|
|
fCapitalization[typeMapPtr->usageTypeEnumValue][0] = intVector[0];
|
|
fCapitalization[typeMapPtr->usageTypeEnumValue][1] = intVector[1];
|
|
}
|
|
}
|
|
}
|
|
tempStatus = U_ZERO_ERROR;
|
|
ures_close(contextTransformUsage);
|
|
}
|
|
ures_close(contextTransforms);
|
|
}
|
|
|
|
tempStatus = U_ZERO_ERROR;
|
|
const LocalPointer<NumberingSystem> numberingSystem(
|
|
NumberingSystem::createInstance(locale, tempStatus), tempStatus);
|
|
if (U_SUCCESS(tempStatus)) {
|
|
// These functions all fail gracefully if passed NULL pointers and
|
|
// do nothing unless U_SUCCESS(tempStatus), so it's only necessary
|
|
// to check for errors once after all calls are made.
|
|
const LocalUResourceBundlePointer numberElementsData(ures_getByKeyWithFallback(
|
|
localeBundle, gNumberElementsTag, NULL, &tempStatus));
|
|
const LocalUResourceBundlePointer nsNameData(ures_getByKeyWithFallback(
|
|
numberElementsData.getAlias(), numberingSystem->getName(), NULL, &tempStatus));
|
|
const LocalUResourceBundlePointer symbolsData(ures_getByKeyWithFallback(
|
|
nsNameData.getAlias(), gSymbolsTag, NULL, &tempStatus));
|
|
fTimeSeparator = ures_getUnicodeStringByKey(
|
|
symbolsData.getAlias(), gTimeSeparatorTag, &tempStatus);
|
|
if (U_FAILURE(tempStatus)) {
|
|
fTimeSeparator.setToBogus();
|
|
}
|
|
}
|
|
|
|
ures_close(localeBundle);
|
|
}
|
|
|
|
if (fTimeSeparator.isBogus()) {
|
|
fTimeSeparator.setTo(DateFormatSymbols::DEFAULT_TIME_SEPARATOR);
|
|
}
|
|
|
|
fWideDayPeriods = loadDayPeriodStrings(calData, gNamesWideTag, FALSE,
|
|
fWideDayPeriodsCount, status);
|
|
fNarrowDayPeriods = loadDayPeriodStrings(calData, gNamesNarrowTag, FALSE,
|
|
fNarrowDayPeriodsCount, status);
|
|
fAbbreviatedDayPeriods = loadDayPeriodStrings(calData, gNamesAbbrTag, FALSE,
|
|
fAbbreviatedDayPeriodsCount, status);
|
|
fStandaloneWideDayPeriods = loadDayPeriodStrings(calData, gNamesWideTag, TRUE,
|
|
fStandaloneWideDayPeriodsCount, status);
|
|
fStandaloneNarrowDayPeriods = loadDayPeriodStrings(calData, gNamesNarrowTag, TRUE,
|
|
fStandaloneNarrowDayPeriodsCount, status);
|
|
fStandaloneAbbreviatedDayPeriods = loadDayPeriodStrings(calData, gNamesAbbrTag, TRUE,
|
|
fStandaloneAbbreviatedDayPeriodsCount, status);
|
|
|
|
UResourceBundle *weekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *abbrWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *shorterWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *narrowWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *standaloneWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *standaloneAbbrWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *standaloneShorterWeekdaysData = NULL; // Data closed by calData
|
|
UResourceBundle *standaloneNarrowWeekdaysData = NULL; // Data closed by calData
|
|
|
|
U_LOCALE_BASED(locBased, *this);
|
|
if (U_FAILURE(status))
|
|
{
|
|
if (useLastResortData)
|
|
{
|
|
// Handle the case in which there is no resource data present.
|
|
// We don't have to generate usable patterns in this situation;
|
|
// we just need to produce something that will be semi-intelligible
|
|
// in most locales.
|
|
|
|
status = U_USING_FALLBACK_WARNING;
|
|
|
|
initField(&fEras, fErasCount, (const UChar *)gLastResortEras, kEraNum, kEraLen, status);
|
|
initField(&fEraNames, fEraNamesCount, (const UChar *)gLastResortEras, kEraNum, kEraLen, status);
|
|
initField(&fNarrowEras, fNarrowErasCount, (const UChar *)gLastResortEras, kEraNum, kEraLen, status);
|
|
initField(&fMonths, fMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fShortMonths, fShortMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fNarrowMonths, fNarrowMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fStandaloneMonths, fStandaloneMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fStandaloneShortMonths, fStandaloneShortMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount, (const UChar *)gLastResortMonthNames, kMonthNum, kMonthLen, status);
|
|
initField(&fWeekdays, fWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fShortWeekdays, fShortWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fShorterWeekdays, fShorterWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fNarrowWeekdays, fNarrowWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fStandaloneWeekdays, fStandaloneWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fStandaloneShortWeekdays, fStandaloneShortWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fStandaloneShorterWeekdays, fStandaloneShorterWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fStandaloneNarrowWeekdays, fStandaloneNarrowWeekdaysCount, (const UChar *)gLastResortDayNames, kDayNum, kDayLen, status);
|
|
initField(&fAmPms, fAmPmsCount, (const UChar *)gLastResortAmPmMarkers, kAmPmNum, kAmPmLen, status);
|
|
initField(&fNarrowAmPms, fNarrowAmPmsCount, (const UChar *)gLastResortAmPmMarkers, kAmPmNum, kAmPmLen, status);
|
|
initField(&fQuarters, fQuartersCount, (const UChar *)gLastResortQuarters, kQuarterNum, kQuarterLen, status);
|
|
initField(&fShortQuarters, fShortQuartersCount, (const UChar *)gLastResortQuarters, kQuarterNum, kQuarterLen, status);
|
|
initField(&fStandaloneQuarters, fStandaloneQuartersCount, (const UChar *)gLastResortQuarters, kQuarterNum, kQuarterLen, status);
|
|
initField(&fStandaloneShortQuarters, fStandaloneShortQuartersCount, (const UChar *)gLastResortQuarters, kQuarterNum, kQuarterLen, status);
|
|
fLocalPatternChars.setTo(TRUE, gPatternChars, PATTERN_CHARS_LEN);
|
|
}
|
|
goto cleanup;
|
|
}
|
|
|
|
// if we make it to here, the resource data is cool, and we can get everything out
|
|
// of it that we need except for the time-zone and localized-pattern data, which
|
|
// are stored in a separate file
|
|
locBased.setLocaleIDs(ures_getLocaleByType(eras, ULOC_VALID_LOCALE, &status),
|
|
ures_getLocaleByType(eras, ULOC_ACTUAL_LOCALE, &status));
|
|
|
|
initField(&fEras, fErasCount, eras, status);
|
|
initField(&fEraNames, fEraNamesCount, eraNames, status);
|
|
initField(&fNarrowEras, fNarrowErasCount, narrowEras, status);
|
|
|
|
initField(&fMonths, fMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesWideTag, status), status);
|
|
initField(&fShortMonths, fShortMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesAbbrTag, status), status);
|
|
|
|
initField(&fNarrowMonths, fNarrowMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesNarrowTag, status), status);
|
|
if(status == U_MISSING_RESOURCE_ERROR) {
|
|
status = U_ZERO_ERROR;
|
|
initField(&fNarrowMonths, fNarrowMonthsCount, calData.getByKey3(gMonthNamesTag, gNamesStandaloneTag, gNamesNarrowTag, status), status);
|
|
}
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { /* If format/narrow not available, use format/abbreviated */
|
|
status = U_ZERO_ERROR;
|
|
initField(&fNarrowMonths, fNarrowMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesAbbrTag, status), status);
|
|
}
|
|
|
|
initField(&fStandaloneMonths, fStandaloneMonthsCount, calData.getByKey3(gMonthNamesTag, gNamesStandaloneTag, gNamesWideTag, status), status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { /* If standalone/wide not available, use format/wide */
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneMonths, fStandaloneMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesWideTag, status), status);
|
|
}
|
|
initField(&fStandaloneShortMonths, fStandaloneShortMonthsCount, calData.getByKey3(gMonthNamesTag, gNamesStandaloneTag, gNamesAbbrTag, status), status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { /* If standalone/abbreviated not available, use format/abbreviated */
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneShortMonths, fStandaloneShortMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesAbbrTag, status), status);
|
|
}
|
|
initField(&fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount, calData.getByKey3(gMonthNamesTag, gNamesStandaloneTag, gNamesNarrowTag, status), status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { /* if standalone/narrow not availabe, try format/narrow */
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesNarrowTag, status), status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) { /* if still not there, use format/abbreviated */
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneNarrowMonths, fStandaloneNarrowMonthsCount, calData.getByKey2(gMonthNamesTag, gNamesAbbrTag, status), status);
|
|
}
|
|
}
|
|
initField(&fAmPms, fAmPmsCount, calData.getByKey(gAmPmMarkersTag, status), status);
|
|
initField(&fNarrowAmPms, fNarrowAmPmsCount, calData.getByKey(gAmPmMarkersNarrowTag, status), status);
|
|
|
|
initField(&fQuarters, fQuartersCount, calData.getByKey2(gQuartersTag, gNamesWideTag, status), status);
|
|
initField(&fShortQuarters, fShortQuartersCount, calData.getByKey2(gQuartersTag, gNamesAbbrTag, status), status);
|
|
|
|
initField(&fStandaloneQuarters, fStandaloneQuartersCount, calData.getByKey3(gQuartersTag, gNamesStandaloneTag, gNamesWideTag, status), status);
|
|
if(status == U_MISSING_RESOURCE_ERROR) {
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneQuarters, fStandaloneQuartersCount, calData.getByKey2(gQuartersTag, gNamesWideTag, status), status);
|
|
}
|
|
|
|
initField(&fStandaloneShortQuarters, fStandaloneShortQuartersCount, calData.getByKey3(gQuartersTag, gNamesStandaloneTag, gNamesAbbrTag, status), status);
|
|
if(status == U_MISSING_RESOURCE_ERROR) {
|
|
status = U_ZERO_ERROR;
|
|
initField(&fStandaloneShortQuarters, fStandaloneShortQuartersCount, calData.getByKey2(gQuartersTag, gNamesAbbrTag, status), status);
|
|
}
|
|
|
|
// ICU 3.8 or later version no longer uses localized date-time pattern characters by default (ticket#5597)
|
|
/*
|
|
// fastCopyFrom()/setTo() - see assignArray comments
|
|
resStr = ures_getStringByKey(fResourceBundle, gLocalPatternCharsTag, &len, &status);
|
|
fLocalPatternChars.setTo(TRUE, resStr, len);
|
|
// If the locale data does not include new pattern chars, use the defaults
|
|
// TODO: Consider making this an error, since this may add conflicting characters.
|
|
if (len < PATTERN_CHARS_LEN) {
|
|
fLocalPatternChars.append(UnicodeString(TRUE, &gPatternChars[len], PATTERN_CHARS_LEN-len));
|
|
}
|
|
*/
|
|
fLocalPatternChars.setTo(TRUE, gPatternChars, PATTERN_CHARS_LEN);
|
|
|
|
// Format wide weekdays -> fWeekdays
|
|
// {sfb} fixed to handle 1-based weekdays
|
|
weekdaysData = calData.getByKey2(gDayNamesTag, gNamesWideTag, status);
|
|
fWeekdaysCount = ures_getSize(weekdaysData);
|
|
fWeekdays = new UnicodeString[fWeekdaysCount+1];
|
|
/* pin the blame on system. If we cannot get a chunk of memory .. the system is dying!*/
|
|
if (fWeekdays == NULL) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fWeekdays[0] empty
|
|
for(i = 0; i<fWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(weekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fWeekdaysCount++;
|
|
|
|
// Format abbreviated weekdays -> fShortWeekdays
|
|
abbrWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
fShortWeekdaysCount = ures_getSize(abbrWeekdaysData);
|
|
fShortWeekdays = new UnicodeString[fShortWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fShortWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fShortWeekdays[0] empty
|
|
for(i = 0; i<fShortWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(abbrWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fShortWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fShortWeekdaysCount++;
|
|
|
|
// Format short weekdays -> fShorterWeekdays (fall back to abbreviated)
|
|
shorterWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesShortTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
shorterWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
}
|
|
fShorterWeekdaysCount = ures_getSize(shorterWeekdaysData);
|
|
fShorterWeekdays = new UnicodeString[fShorterWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fShorterWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fShorterWeekdays[0] empty
|
|
for(i = 0; i<fShorterWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(shorterWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fShorterWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fShorterWeekdaysCount++;
|
|
|
|
// Format narrow weekdays -> fNarrowWeekdays
|
|
narrowWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesNarrowTag, status);
|
|
if(status == U_MISSING_RESOURCE_ERROR) {
|
|
status = U_ZERO_ERROR;
|
|
narrowWeekdaysData = calData.getByKey3(gDayNamesTag, gNamesStandaloneTag, gNamesNarrowTag, status);
|
|
}
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
narrowWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
}
|
|
fNarrowWeekdaysCount = ures_getSize(narrowWeekdaysData);
|
|
fNarrowWeekdays = new UnicodeString[fNarrowWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fNarrowWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fNarrowWeekdays[0] empty
|
|
for(i = 0; i<fNarrowWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(narrowWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fNarrowWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fNarrowWeekdaysCount++;
|
|
|
|
// Stand-alone wide weekdays -> fStandaloneWeekdays
|
|
standaloneWeekdaysData = calData.getByKey3(gDayNamesTag, gNamesStandaloneTag, gNamesWideTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
standaloneWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesWideTag, status);
|
|
}
|
|
fStandaloneWeekdaysCount = ures_getSize(standaloneWeekdaysData);
|
|
fStandaloneWeekdays = new UnicodeString[fStandaloneWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fStandaloneWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fStandaloneWeekdays[0] empty
|
|
for(i = 0; i<fStandaloneWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(standaloneWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fStandaloneWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fStandaloneWeekdaysCount++;
|
|
|
|
// Stand-alone abbreviated weekdays -> fStandaloneShortWeekdays
|
|
standaloneAbbrWeekdaysData = calData.getByKey3(gDayNamesTag, gNamesStandaloneTag, gNamesAbbrTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
standaloneAbbrWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
}
|
|
fStandaloneShortWeekdaysCount = ures_getSize(standaloneAbbrWeekdaysData);
|
|
fStandaloneShortWeekdays = new UnicodeString[fStandaloneShortWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fStandaloneShortWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fStandaloneShortWeekdays[0] empty
|
|
for(i = 0; i<fStandaloneShortWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(standaloneAbbrWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fStandaloneShortWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fStandaloneShortWeekdaysCount++;
|
|
|
|
// Stand-alone short weekdays -> fStandaloneShorterWeekdays (fall back to format abbreviated)
|
|
standaloneShorterWeekdaysData = calData.getByKey3(gDayNamesTag, gNamesStandaloneTag, gNamesShortTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
standaloneShorterWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
}
|
|
fStandaloneShorterWeekdaysCount = ures_getSize(standaloneShorterWeekdaysData);
|
|
fStandaloneShorterWeekdays = new UnicodeString[fStandaloneShorterWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fStandaloneShorterWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fStandaloneShorterWeekdays[0] empty
|
|
for(i = 0; i<fStandaloneShorterWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(standaloneShorterWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fStandaloneShorterWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fStandaloneShorterWeekdaysCount++;
|
|
|
|
// Stand-alone narrow weekdays -> fStandaloneNarrowWeekdays
|
|
standaloneNarrowWeekdaysData = calData.getByKey3(gDayNamesTag, gNamesStandaloneTag, gNamesNarrowTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
standaloneNarrowWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesNarrowTag, status);
|
|
if ( status == U_MISSING_RESOURCE_ERROR ) {
|
|
status = U_ZERO_ERROR;
|
|
standaloneNarrowWeekdaysData = calData.getByKey2(gDayNamesTag, gNamesAbbrTag, status);
|
|
}
|
|
}
|
|
fStandaloneNarrowWeekdaysCount = ures_getSize(standaloneNarrowWeekdaysData);
|
|
fStandaloneNarrowWeekdays = new UnicodeString[fStandaloneNarrowWeekdaysCount+1];
|
|
/* test for NULL */
|
|
if (fStandaloneNarrowWeekdays == 0) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
goto cleanup;
|
|
}
|
|
// leave fStandaloneNarrowWeekdays[0] empty
|
|
for(i = 0; i<fStandaloneNarrowWeekdaysCount; i++) {
|
|
resStr = ures_getStringByIndex(standaloneNarrowWeekdaysData, i, &len, &status);
|
|
// setTo() - see assignArray comments
|
|
fStandaloneNarrowWeekdays[i+1].setTo(TRUE, resStr, len);
|
|
}
|
|
fStandaloneNarrowWeekdaysCount++;
|
|
|
|
cleanup:
|
|
ures_close(eras);
|
|
ures_close(eraNames);
|
|
ures_close(narrowEras);
|
|
}
|
|
|
|
Locale
|
|
DateFormatSymbols::getLocale(ULocDataLocaleType type, UErrorCode& status) const {
|
|
U_LOCALE_BASED(locBased, *this);
|
|
return locBased.getLocale(type, status);
|
|
}
|
|
|
|
U_NAMESPACE_END
|
|
|
|
#endif /* #if !UCONFIG_NO_FORMATTING */
|
|
|
|
//eof
|