2013-11-22 18:26:22 +00:00
|
|
|
/*
|
2014-01-11 00:30:39 +00:00
|
|
|
******************************************************************************
|
|
|
|
* Copyright (C) 2014, International Business Machines Corporation and
|
2013-11-22 18:26:22 +00:00
|
|
|
* others. All Rights Reserved.
|
2014-01-11 00:30:39 +00:00
|
|
|
******************************************************************************
|
2013-11-22 18:26:22 +00:00
|
|
|
*
|
|
|
|
* File RELDATEFMT.CPP
|
2014-01-11 00:30:39 +00:00
|
|
|
******************************************************************************
|
2013-11-22 18:26:22 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "unicode/reldatefmt.h"
|
|
|
|
|
2013-12-03 04:48:00 +00:00
|
|
|
#if !UCONFIG_NO_FORMATTING
|
|
|
|
|
2013-11-22 18:26:22 +00:00
|
|
|
#include "unicode/localpointer.h"
|
|
|
|
#include "unicode/plurrule.h"
|
|
|
|
#include "unicode/msgfmt.h"
|
|
|
|
#include "unicode/decimfmt.h"
|
|
|
|
#include "unicode/numfmt.h"
|
|
|
|
#include "lrucache.h"
|
|
|
|
#include "uresimp.h"
|
|
|
|
#include "unicode/ures.h"
|
|
|
|
#include "cstring.h"
|
|
|
|
#include "plurrule_impl.h"
|
|
|
|
#include "ucln_in.h"
|
2013-11-22 23:08:24 +00:00
|
|
|
#include "mutex.h"
|
2014-01-11 00:30:39 +00:00
|
|
|
#include "charstr.h"
|
2013-11-22 18:26:22 +00:00
|
|
|
|
|
|
|
#include "sharedptr.h"
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
// Copied from uscript_props.cpp
|
|
|
|
#define LENGTHOF(array) (int32_t)(sizeof(array)/sizeof((array)[0]))
|
|
|
|
|
2013-11-22 18:26:22 +00:00
|
|
|
static icu::LRUCache *gCache = NULL;
|
|
|
|
static UMutex gCacheMutex = U_MUTEX_INITIALIZER;
|
|
|
|
static icu::UInitOnce gCacheInitOnce = U_INITONCE_INITIALIZER;
|
|
|
|
|
|
|
|
U_CDECL_BEGIN
|
|
|
|
static UBool U_CALLCONV reldatefmt_cleanup() {
|
|
|
|
gCacheInitOnce.reset();
|
|
|
|
if (gCache) {
|
|
|
|
delete gCache;
|
|
|
|
gCache = NULL;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
U_CDECL_END
|
|
|
|
|
|
|
|
U_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
// other must always be first.
|
|
|
|
static const char * const gPluralForms[] = {
|
2014-01-11 00:30:39 +00:00
|
|
|
"other", "zero", "one", "two", "few", "many"};
|
2013-11-22 18:26:22 +00:00
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static const UChar gPlaceholder[] = { 0x7b, 0x30, 0x7d}; /* {0} */
|
2013-11-22 18:26:22 +00:00
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
class QualitativeUnits : public UMemory {
|
2013-11-22 18:26:22 +00:00
|
|
|
public:
|
|
|
|
QualitativeUnits() { }
|
2013-12-04 20:46:08 +00:00
|
|
|
UnicodeString data[UDAT_ABSOLUTE_UNIT_COUNT][UDAT_DIRECTION_COUNT];
|
2013-11-22 18:26:22 +00:00
|
|
|
private:
|
|
|
|
QualitativeUnits(const QualitativeUnits &other);
|
|
|
|
QualitativeUnits &operator=(const QualitativeUnits& other);
|
|
|
|
};
|
|
|
|
|
2013-12-17 21:02:16 +00:00
|
|
|
struct UnitPattern {
|
2014-01-11 00:30:39 +00:00
|
|
|
UnicodeString pattern;
|
|
|
|
int32_t offset; // Offset of "{0}". -1 means no {0}.
|
|
|
|
UBool valid; // True if initialize, false otherwise.
|
2013-12-17 21:02:16 +00:00
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
UnitPattern() : pattern(), offset(0), valid(FALSE) {
|
2013-12-17 21:02:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void set(const UnicodeString &patternStr) {
|
2014-01-11 00:30:39 +00:00
|
|
|
pattern = patternStr;
|
|
|
|
offset = patternStr.indexOf(gPlaceholder, LENGTHOF(gPlaceholder), 0);
|
2013-12-17 21:02:16 +00:00
|
|
|
valid = TRUE;
|
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
UnicodeString& append(
|
|
|
|
double quantity,
|
|
|
|
const NumberFormat &nf,
|
|
|
|
UnicodeString &appendTo) const {
|
2013-12-17 21:02:16 +00:00
|
|
|
if (!valid) {
|
|
|
|
return appendTo;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
if (offset == -1) {
|
|
|
|
appendTo.append(pattern);
|
|
|
|
return appendTo;
|
2013-12-17 21:02:16 +00:00
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
appendTo.append(pattern.tempSubStringBetween(0, offset));
|
|
|
|
nf.format(quantity, appendTo);
|
|
|
|
appendTo.append(pattern.tempSubStringBetween(
|
|
|
|
offset + LENGTHOF(gPlaceholder)));
|
2013-12-17 21:02:16 +00:00
|
|
|
return appendTo;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
class QuantitativeUnits : public UMemory {
|
2013-11-22 18:26:22 +00:00
|
|
|
public:
|
|
|
|
QuantitativeUnits() { }
|
2014-01-11 00:30:39 +00:00
|
|
|
UnitPattern data[UDAT_RELATIVE_UNIT_COUNT][2][LENGTHOF(gPluralForms)];
|
2013-11-22 18:26:22 +00:00
|
|
|
private:
|
|
|
|
QuantitativeUnits(const QuantitativeUnits &other);
|
|
|
|
QuantitativeUnits &operator=(const QuantitativeUnits& other);
|
|
|
|
};
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
class RelativeDateTimeData : public SharedObject {
|
2013-11-22 18:26:22 +00:00
|
|
|
public:
|
|
|
|
SharedPtr<QualitativeUnits> qualitativeUnits;
|
|
|
|
SharedPtr<QuantitativeUnits> quantitativeUnits;
|
|
|
|
SharedPtr<MessageFormat> combinedDateAndTime;
|
|
|
|
SharedPtr<PluralRules> pluralRules;
|
|
|
|
SharedPtr<NumberFormat> numberFormat;
|
2014-01-11 00:30:39 +00:00
|
|
|
virtual ~RelativeDateTimeData();
|
2013-11-22 18:26:22 +00:00
|
|
|
private:
|
|
|
|
RelativeDateTimeData &operator=(const RelativeDateTimeData& other);
|
|
|
|
};
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeData::~RelativeDateTimeData() {
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getStringWithFallback(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *key,
|
|
|
|
UnicodeString &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
int32_t len = 0;
|
|
|
|
const UChar *resStr = ures_getStringByKeyWithFallback(
|
|
|
|
resource, key, &len, &status);
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
result.setTo(TRUE, resStr, len);
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getOptionalStringWithFallback(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *key,
|
|
|
|
UnicodeString &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
int32_t len = 0;
|
|
|
|
const UChar *resStr = ures_getStringByKey(
|
|
|
|
resource, key, &len, &status);
|
|
|
|
if (status == U_MISSING_RESOURCE_ERROR) {
|
|
|
|
result.remove();
|
|
|
|
status = U_ZERO_ERROR;
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
result.setTo(TRUE, resStr, len);
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getString(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
UnicodeString &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
int32_t len = 0;
|
|
|
|
const UChar *resStr = ures_getString(resource, &len, &status);
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
result.setTo(TRUE, resStr, len);
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getUnitPattern(
|
2013-12-17 21:02:16 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
UnitPattern &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-12-17 21:02:16 +00:00
|
|
|
}
|
|
|
|
UnicodeString rawPattern;
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getString(resource, rawPattern, status)) {
|
|
|
|
return FALSE;
|
2013-12-17 21:02:16 +00:00
|
|
|
}
|
|
|
|
result.set(rawPattern);
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-12-17 21:02:16 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getStringByIndex(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
int32_t idx,
|
|
|
|
UnicodeString &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
int32_t len = 0;
|
|
|
|
const UChar *resStr = ures_getStringByIndex(
|
|
|
|
resource, idx, &len, &status);
|
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
result.setTo(TRUE, resStr, len);
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void addQualitativeUnit(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
UDateAbsoluteUnit absoluteUnit,
|
|
|
|
const UnicodeString &unitName,
|
|
|
|
QualitativeUnits &qualitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
getStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"-1",
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_LAST],
|
|
|
|
status);
|
|
|
|
getStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"0",
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_THIS],
|
|
|
|
status);
|
|
|
|
getStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"1",
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_NEXT],
|
|
|
|
status);
|
|
|
|
getOptionalStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"-2",
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_LAST_2],
|
|
|
|
status);
|
|
|
|
getOptionalStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"2",
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_NEXT_2],
|
|
|
|
status);
|
|
|
|
qualitativeUnits.data[absoluteUnit][UDAT_DIRECTION_PLAIN] = unitName;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int32_t getPluralIndex(const char *pluralForm) {
|
2014-01-11 00:30:39 +00:00
|
|
|
int32_t len = LENGTHOF(gPluralForms);
|
|
|
|
for (int32_t i = 0; i < len; ++i) {
|
2013-11-22 18:26:22 +00:00
|
|
|
if (uprv_strcmp(pluralForm, gPluralForms[i]) == 0) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addTimeUnit(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
UDateRelativeUnit relativeUnit,
|
|
|
|
int32_t pastOrFuture,
|
|
|
|
QuantitativeUnits &quantitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int32_t size = ures_getSize(resource);
|
|
|
|
for (int32_t i = 0; i < size; ++i) {
|
|
|
|
LocalUResourceBundlePointer pluralBundle(
|
|
|
|
ures_getByIndex(resource, i, NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int32_t pluralIndex = getPluralIndex(
|
|
|
|
ures_getKey(pluralBundle.getAlias()));
|
|
|
|
if (pluralIndex != -1) {
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getUnitPattern(
|
2013-11-22 18:26:22 +00:00
|
|
|
pluralBundle.getAlias(),
|
|
|
|
quantitativeUnits.data[relativeUnit][pastOrFuture][pluralIndex],
|
2014-01-11 00:30:39 +00:00
|
|
|
status)) {
|
2013-11-22 18:26:22 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addTimeUnit(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
UDateRelativeUnit relativeUnit,
|
|
|
|
QuantitativeUnits &quantitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
LocalUResourceBundlePointer topLevel(
|
2014-01-11 00:30:39 +00:00
|
|
|
ures_getByKeyWithFallback(
|
|
|
|
resource, "relativeTime", NULL, &status));
|
2013-11-22 18:26:22 +00:00
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
LocalUResourceBundlePointer futureBundle(ures_getByKeyWithFallback(
|
|
|
|
topLevel.getAlias(), "future", NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addTimeUnit(
|
|
|
|
futureBundle.getAlias(),
|
|
|
|
relativeUnit,
|
|
|
|
1,
|
|
|
|
quantitativeUnits,
|
|
|
|
status);
|
|
|
|
LocalUResourceBundlePointer pastBundle(ures_getByKeyWithFallback(
|
|
|
|
topLevel.getAlias(), "past", NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addTimeUnit(
|
|
|
|
pastBundle.getAlias(),
|
|
|
|
relativeUnit,
|
|
|
|
0,
|
|
|
|
quantitativeUnits,
|
|
|
|
status);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addTimeUnit(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *path,
|
|
|
|
UDateRelativeUnit relativeUnit,
|
|
|
|
QuantitativeUnits &quantitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
LocalUResourceBundlePointer topLevel(
|
|
|
|
ures_getByKeyWithFallback(resource, path, NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addTimeUnit(topLevel.getAlias(), relativeUnit, quantitativeUnits, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addTimeUnit(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *path,
|
|
|
|
UDateRelativeUnit relativeUnit,
|
|
|
|
UDateAbsoluteUnit absoluteUnit,
|
|
|
|
QuantitativeUnits &quantitativeUnits,
|
|
|
|
QualitativeUnits &qualitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
LocalUResourceBundlePointer topLevel(
|
|
|
|
ures_getByKeyWithFallback(resource, path, NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addTimeUnit(topLevel.getAlias(), relativeUnit, quantitativeUnits, status);
|
|
|
|
UnicodeString unitName;
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getStringWithFallback(topLevel.getAlias(), "dn", unitName, status)) {
|
2013-11-22 18:26:22 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO(Travis Keep): This is a hack to get around CLDR bug 6818.
|
|
|
|
const char *localeId = ures_getLocaleByType(
|
|
|
|
topLevel.getAlias(), ULOC_ACTUAL_LOCALE, &status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Locale locale(localeId);
|
|
|
|
if (uprv_strcmp("en", locale.getLanguage()) == 0) {
|
|
|
|
unitName.toLower();
|
|
|
|
}
|
|
|
|
// end hack
|
|
|
|
ures_getByKeyWithFallback(
|
|
|
|
topLevel.getAlias(), "relative", topLevel.getAlias(), &status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addQualitativeUnit(
|
|
|
|
topLevel.getAlias(),
|
|
|
|
absoluteUnit,
|
|
|
|
unitName,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void readDaysOfWeek(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *path,
|
|
|
|
UnicodeString *daysOfWeek,
|
|
|
|
UErrorCode &status) {
|
|
|
|
LocalUResourceBundlePointer topLevel(
|
|
|
|
ures_getByKeyWithFallback(resource, path, NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int32_t size = ures_getSize(topLevel.getAlias());
|
|
|
|
if (size != 7) {
|
|
|
|
status = U_INTERNAL_PROGRAM_ERROR;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (int32_t i = 0; i < size; ++i) {
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getStringByIndex(topLevel.getAlias(), i, daysOfWeek[i], status)) {
|
2013-11-22 18:26:22 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addWeekDay(
|
|
|
|
const UResourceBundle *resource,
|
|
|
|
const char *path,
|
|
|
|
const UnicodeString *daysOfWeek,
|
|
|
|
UDateAbsoluteUnit absoluteUnit,
|
|
|
|
QualitativeUnits &qualitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
LocalUResourceBundlePointer topLevel(
|
|
|
|
ures_getByKeyWithFallback(resource, path, NULL, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
addQualitativeUnit(
|
|
|
|
topLevel.getAlias(),
|
|
|
|
absoluteUnit,
|
|
|
|
daysOfWeek[absoluteUnit - UDAT_ABSOLUTE_SUNDAY],
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool load(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
QualitativeUnits &qualitativeUnits,
|
|
|
|
QuantitativeUnits &quantitativeUnits,
|
|
|
|
UErrorCode &status) {
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/day",
|
|
|
|
UDAT_RELATIVE_DAYS,
|
|
|
|
UDAT_ABSOLUTE_DAY,
|
|
|
|
quantitativeUnits,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/week",
|
|
|
|
UDAT_RELATIVE_WEEKS,
|
|
|
|
UDAT_ABSOLUTE_WEEK,
|
|
|
|
quantitativeUnits,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/month",
|
|
|
|
UDAT_RELATIVE_MONTHS,
|
|
|
|
UDAT_ABSOLUTE_MONTH,
|
|
|
|
quantitativeUnits,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/year",
|
|
|
|
UDAT_RELATIVE_YEARS,
|
|
|
|
UDAT_ABSOLUTE_YEAR,
|
|
|
|
quantitativeUnits,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/second",
|
|
|
|
UDAT_RELATIVE_SECONDS,
|
|
|
|
quantitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/minute",
|
|
|
|
UDAT_RELATIVE_MINUTES,
|
|
|
|
quantitativeUnits,
|
|
|
|
status);
|
|
|
|
addTimeUnit(
|
|
|
|
resource,
|
|
|
|
"fields/hour",
|
|
|
|
UDAT_RELATIVE_HOURS,
|
|
|
|
quantitativeUnits,
|
|
|
|
status);
|
|
|
|
getStringWithFallback(
|
|
|
|
resource,
|
|
|
|
"fields/second/relative/0",
|
|
|
|
qualitativeUnits.data[UDAT_ABSOLUTE_NOW][UDAT_DIRECTION_PLAIN],
|
|
|
|
status);
|
|
|
|
UnicodeString daysOfWeek[7];
|
|
|
|
readDaysOfWeek(
|
|
|
|
resource,
|
|
|
|
"calendar/gregorian/dayNames/stand-alone/wide",
|
|
|
|
daysOfWeek,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/mon/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_MONDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/tue/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_TUESDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/wed/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_WEDNESDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/thu/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_THURSDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/fri/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_FRIDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/sat/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_SATURDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
|
|
|
addWeekDay(
|
|
|
|
resource,
|
|
|
|
"fields/sun/relative",
|
|
|
|
daysOfWeek,
|
|
|
|
UDAT_ABSOLUTE_SUNDAY,
|
|
|
|
qualitativeUnits,
|
|
|
|
status);
|
2014-01-11 00:30:39 +00:00
|
|
|
return U_SUCCESS(status);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static UBool getDateTimePattern(
|
2013-11-22 18:26:22 +00:00
|
|
|
const UResourceBundle *resource,
|
|
|
|
UnicodeString &result,
|
|
|
|
UErrorCode &status) {
|
|
|
|
UnicodeString defaultCalendarName;
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getStringWithFallback(
|
2013-11-22 18:26:22 +00:00
|
|
|
resource,
|
|
|
|
"calendar/default",
|
|
|
|
defaultCalendarName,
|
2014-01-11 00:30:39 +00:00
|
|
|
status)) {
|
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
CharString pathBuffer;
|
|
|
|
pathBuffer.append("calendar/", status)
|
|
|
|
.appendInvariantChars(defaultCalendarName, status)
|
|
|
|
.append("/DateTimePatterns", status);
|
2013-11-22 18:26:22 +00:00
|
|
|
LocalUResourceBundlePointer topLevel(
|
2014-01-11 00:30:39 +00:00
|
|
|
ures_getByKeyWithFallback(
|
|
|
|
resource, pathBuffer.data(), NULL, &status));
|
2013-11-22 18:26:22 +00:00
|
|
|
if (U_FAILURE(status)) {
|
2014-01-11 00:30:39 +00:00
|
|
|
return FALSE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
int32_t size = ures_getSize(topLevel.getAlias());
|
2014-01-11 00:30:39 +00:00
|
|
|
if (size <= 8) {
|
2013-11-22 18:26:22 +00:00
|
|
|
// Oops, size is to small to access the index that we want, fallback
|
|
|
|
// to a hard-coded value.
|
2013-12-03 23:27:49 +00:00
|
|
|
result = UNICODE_STRING_SIMPLE("{1} {0}");
|
2014-01-11 00:30:39 +00:00
|
|
|
return TRUE;
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
return getStringByIndex(topLevel.getAlias(), 8, result, status);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static SharedObject *U_CALLCONV createData(
|
|
|
|
const char *localeId, UErrorCode &status) {
|
2013-11-22 18:26:22 +00:00
|
|
|
LocalUResourceBundlePointer topLevel(ures_open(NULL, localeId, &status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
LocalPointer<RelativeDateTimeData> result(new RelativeDateTimeData());
|
|
|
|
LocalPointer<QualitativeUnits> qualitativeUnits(new QualitativeUnits());
|
|
|
|
LocalPointer<QuantitativeUnits> quantitativeUnits(new QuantitativeUnits());
|
2014-01-11 00:30:39 +00:00
|
|
|
if (result.getAlias() == NULL
|
|
|
|
|| qualitativeUnits.getAlias() == NULL
|
|
|
|
|| quantitativeUnits.getAlias() == NULL) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!load(
|
|
|
|
topLevel.getAlias(),
|
|
|
|
*qualitativeUnits,
|
|
|
|
*quantitativeUnits,
|
|
|
|
status)) {
|
2013-11-22 18:26:22 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!result->qualitativeUnits.reset(qualitativeUnits.orphan())) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!result->quantitativeUnits.reset(quantitativeUnits.orphan())) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UnicodeString dateTimePattern;
|
2014-01-11 00:30:39 +00:00
|
|
|
if (!getDateTimePattern(topLevel.getAlias(), dateTimePattern, status)) {
|
2013-11-22 18:26:22 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
LocalPointer<MessageFormat> mf(
|
|
|
|
new MessageFormat(dateTimePattern, localeId, status));
|
2013-11-22 18:26:22 +00:00
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
if (mf.getAlias() == NULL) {
|
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!result->combinedDateAndTime.reset(mf.orphan())) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
LocalPointer<PluralRules> pr(PluralRules::forLocale(localeId, status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!result->pluralRules.reset(pr.orphan())) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
LocalPointer<NumberFormat> nf(
|
|
|
|
NumberFormat::createInstance(localeId, status));
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!result->numberFormat.reset(nf.orphan())) {
|
2013-11-22 18:26:22 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return result.orphan();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void U_CALLCONV cacheInit(UErrorCode &status) {
|
|
|
|
U_ASSERT(gCache == NULL);
|
|
|
|
ucln_i18n_registerCleanup(UCLN_I18N_RELDATEFMT, reldatefmt_cleanup);
|
2013-11-22 23:08:24 +00:00
|
|
|
gCache = new SimpleLRUCache(100, &createData, status);
|
2013-11-22 18:26:22 +00:00
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
delete gCache;
|
|
|
|
gCache = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
static void getFromCache(
|
|
|
|
const char *locale,
|
|
|
|
const RelativeDateTimeData *&ptr,
|
|
|
|
UErrorCode &status) {
|
2013-11-22 18:26:22 +00:00
|
|
|
umtx_initOnce(gCacheInitOnce, &cacheInit, status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
2013-11-22 23:08:24 +00:00
|
|
|
Mutex lock(&gCacheMutex);
|
2013-11-22 18:26:22 +00:00
|
|
|
gCache->get(locale, ptr, status);
|
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeFormatter::RelativeDateTimeFormatter(UErrorCode& status)
|
|
|
|
: ptr(NULL) {
|
2013-12-04 22:50:25 +00:00
|
|
|
getFromCache(Locale::getDefault().getName(), ptr, status);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeFormatter::RelativeDateTimeFormatter(
|
|
|
|
const Locale& locale, UErrorCode& status) : ptr(NULL) {
|
2013-11-22 18:26:22 +00:00
|
|
|
getFromCache(locale.getName(), ptr, status);
|
|
|
|
}
|
|
|
|
|
2013-12-04 22:50:25 +00:00
|
|
|
RelativeDateTimeFormatter::RelativeDateTimeFormatter(
|
2014-01-11 00:30:39 +00:00
|
|
|
const Locale& locale, NumberFormat *nfToAdopt, UErrorCode& status)
|
|
|
|
: ptr(NULL) {
|
2013-12-04 22:50:25 +00:00
|
|
|
getFromCache(locale.getName(), ptr, status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeData* wptr = SharedObject::copyOnWrite(ptr);
|
2013-12-04 22:50:25 +00:00
|
|
|
if (wptr == NULL) {
|
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return;
|
|
|
|
}
|
2014-01-14 22:39:13 +00:00
|
|
|
if (!wptr->numberFormat.reset(nfToAdopt)) {
|
2013-12-04 22:50:25 +00:00
|
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeFormatter::RelativeDateTimeFormatter(
|
|
|
|
const RelativeDateTimeFormatter& other) : ptr(other.ptr) {
|
|
|
|
ptr->addRef();
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
RelativeDateTimeFormatter& RelativeDateTimeFormatter::operator=(
|
|
|
|
const RelativeDateTimeFormatter& other) {
|
2013-11-22 18:26:22 +00:00
|
|
|
if (this != &other) {
|
2014-01-11 00:30:39 +00:00
|
|
|
SharedObject::copyPtr(other.ptr, ptr);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
RelativeDateTimeFormatter::~RelativeDateTimeFormatter() {
|
2014-01-11 00:30:39 +00:00
|
|
|
ptr->removeRef();
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 00:30:39 +00:00
|
|
|
const NumberFormat& RelativeDateTimeFormatter::getNumberFormat() const {
|
|
|
|
return *ptr->numberFormat;
|
|
|
|
}
|
2013-11-22 18:26:22 +00:00
|
|
|
|
|
|
|
UnicodeString& RelativeDateTimeFormatter::format(
|
|
|
|
double quantity, UDateDirection direction, UDateRelativeUnit unit,
|
|
|
|
UnicodeString& appendTo, UErrorCode& status) const {
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return appendTo;
|
|
|
|
}
|
|
|
|
if (direction != UDAT_DIRECTION_LAST && direction != UDAT_DIRECTION_NEXT) {
|
|
|
|
status = U_ILLEGAL_ARGUMENT_ERROR;
|
|
|
|
return appendTo;
|
|
|
|
}
|
|
|
|
FixedDecimal dec(quantity);
|
|
|
|
const DecimalFormat *decFmt = dynamic_cast<const DecimalFormat *>(
|
|
|
|
ptr->numberFormat.readOnly());
|
|
|
|
if (decFmt != NULL) {
|
|
|
|
dec = decFmt->getFixedDecimal(quantity, status);
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
CharString buffer;
|
|
|
|
buffer.appendInvariantChars(ptr->pluralRules->select(dec), status);
|
2013-11-22 18:26:22 +00:00
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return appendTo;
|
|
|
|
}
|
2014-01-11 00:30:39 +00:00
|
|
|
|
|
|
|
int32_t pluralIndex = getPluralIndex(buffer.data());
|
2013-11-22 18:26:22 +00:00
|
|
|
if (pluralIndex == -1) {
|
|
|
|
pluralIndex = 0;
|
|
|
|
}
|
|
|
|
int32_t bFuture = direction == UDAT_DIRECTION_NEXT ? 1 : 0;
|
2013-12-17 21:02:16 +00:00
|
|
|
const UnitPattern *pattern = &ptr->quantitativeUnits->data[unit][bFuture][pluralIndex];
|
|
|
|
if (!pattern->valid) {
|
2013-11-22 18:26:22 +00:00
|
|
|
pattern = &ptr->quantitativeUnits->data[unit][bFuture][0];
|
|
|
|
}
|
2013-12-17 21:02:16 +00:00
|
|
|
return pattern->append(quantity, *ptr->numberFormat, appendTo);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeString& RelativeDateTimeFormatter::format(
|
|
|
|
UDateDirection direction, UDateAbsoluteUnit unit,
|
|
|
|
UnicodeString& appendTo, UErrorCode& status) const {
|
|
|
|
if (U_FAILURE(status)) {
|
|
|
|
return appendTo;
|
|
|
|
}
|
|
|
|
if (unit == UDAT_ABSOLUTE_NOW && direction != UDAT_DIRECTION_PLAIN) {
|
|
|
|
status = U_ILLEGAL_ARGUMENT_ERROR;
|
|
|
|
return appendTo;
|
|
|
|
}
|
|
|
|
return appendTo.append(ptr->qualitativeUnits->data[unit][direction]);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeString& RelativeDateTimeFormatter::combineDateAndTime(
|
|
|
|
const UnicodeString& relativeDateString, const UnicodeString& timeString,
|
|
|
|
UnicodeString& appendTo, UErrorCode& status) const {
|
2014-01-11 00:30:39 +00:00
|
|
|
Formattable args[2] = {timeString, relativeDateString};
|
2013-11-22 18:26:22 +00:00
|
|
|
FieldPosition fpos(0);
|
2014-01-11 00:30:39 +00:00
|
|
|
return ptr->combinedDateAndTime->format(args, 2, appendTo, fpos, status);
|
2013-11-22 18:26:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
U_NAMESPACE_END
|
|
|
|
|
2013-12-03 04:48:00 +00:00
|
|
|
#endif /* !UCONFIG_NO_FORMATTING */
|
|
|
|
|