scuffed-code/icu4c/source/i18n/quantityformatter.cpp

218 lines
6.0 KiB
C++
Raw Normal View History

/*
******************************************************************************
* Copyright (C) 2014-2015, International Business Machines
* Corporation and others. All Rights Reserved.
******************************************************************************
* quantityformatter.cpp
*/
#include "quantityformatter.h"
#include "simplepatternformatter.h"
#include "uassert.h"
#include "unicode/unistr.h"
#include "unicode/decimfmt.h"
#include "cstring.h"
#include "unicode/plurrule.h"
#include "charstr.h"
#include "unicode/fmtable.h"
#include "unicode/fieldpos.h"
#include "visibledigits.h"
#if !UCONFIG_NO_FORMATTING
U_NAMESPACE_BEGIN
/**
* Plural forms in index order: "other", "zero", "one", "two", "few", "many"
* "other" must be first.
*/
static int32_t getPluralIndex(const char *pluralForm) {
switch (*pluralForm++) {
case 'f':
if (uprv_strcmp(pluralForm, "ew") == 0) {
return 4;
}
case 'm':
if (uprv_strcmp(pluralForm, "any") == 0) {
return 5;
}
case 'o':
if (uprv_strcmp(pluralForm, "ther") == 0) {
return 0;
} else if (uprv_strcmp(pluralForm, "ne") == 0) {
return 2;
}
break;
case 't':
if (uprv_strcmp(pluralForm, "wo") == 0) {
return 3;
}
case 'z':
if (uprv_strcmp(pluralForm, "ero") == 0) {
return 1;
}
default:
break;
}
return -1;
}
QuantityFormatter::QuantityFormatter() {
for (int32_t i = 0; i < UPRV_LENGTHOF(formatters); ++i) {
formatters[i] = NULL;
}
}
QuantityFormatter::QuantityFormatter(const QuantityFormatter &other) {
for (int32_t i = 0; i < UPRV_LENGTHOF(formatters); ++i) {
if (other.formatters[i] == NULL) {
formatters[i] = NULL;
} else {
formatters[i] = new SimplePatternFormatter(*other.formatters[i]);
}
}
}
QuantityFormatter &QuantityFormatter::operator=(
const QuantityFormatter& other) {
if (this == &other) {
return *this;
}
for (int32_t i = 0; i < UPRV_LENGTHOF(formatters); ++i) {
delete formatters[i];
if (other.formatters[i] == NULL) {
formatters[i] = NULL;
} else {
formatters[i] = new SimplePatternFormatter(*other.formatters[i]);
}
}
return *this;
}
QuantityFormatter::~QuantityFormatter() {
for (int32_t i = 0; i < UPRV_LENGTHOF(formatters); ++i) {
delete formatters[i];
}
}
void QuantityFormatter::reset() {
for (int32_t i = 0; i < UPRV_LENGTHOF(formatters); ++i) {
delete formatters[i];
formatters[i] = NULL;
}
}
UBool QuantityFormatter::add(
const char *variant,
const UnicodeString &rawPattern,
UErrorCode &status) {
if (U_FAILURE(status)) {
return FALSE;
}
int32_t pluralIndex = getPluralIndex(variant);
if (pluralIndex == -1) {
status = U_ILLEGAL_ARGUMENT_ERROR;
return FALSE;
}
SimplePatternFormatter *newFmt =
new SimplePatternFormatter(rawPattern);
if (newFmt == NULL) {
status = U_MEMORY_ALLOCATION_ERROR;
return FALSE;
}
if (newFmt->getPlaceholderCount() > 1) {
delete newFmt;
status = U_ILLEGAL_ARGUMENT_ERROR;
return FALSE;
}
delete formatters[pluralIndex];
formatters[pluralIndex] = newFmt;
return TRUE;
}
UBool QuantityFormatter::isValid() const {
return formatters[0] != NULL;
}
const SimplePatternFormatter *QuantityFormatter::getByVariant(
const char *variant) const {
int32_t pluralIndex = getPluralIndex(variant);
if (pluralIndex == -1) {
pluralIndex = 0;
}
const SimplePatternFormatter *pattern = formatters[pluralIndex];
if (pattern == NULL) {
pattern = formatters[0];
}
return pattern;
}
UnicodeString &QuantityFormatter::format(
const Formattable& quantity,
const NumberFormat &fmt,
const PluralRules &rules,
UnicodeString &appendTo,
FieldPosition &pos,
UErrorCode &status) const {
if (U_FAILURE(status)) {
return appendTo;
}
UnicodeString count;
VisibleDigitsWithExponent digits;
const DecimalFormat *decFmt = dynamic_cast<const DecimalFormat *>(&fmt);
if (decFmt != NULL) {
decFmt->initVisibleDigitsWithExponent(quantity, digits, status);
if (U_FAILURE(status)) {
return appendTo;
}
count = rules.select(digits);
} else {
if (quantity.getType() == Formattable::kDouble) {
count = rules.select(quantity.getDouble());
} else if (quantity.getType() == Formattable::kLong) {
count = rules.select(quantity.getLong());
} else if (quantity.getType() == Formattable::kInt64) {
count = rules.select((double) quantity.getInt64());
} else {
status = U_ILLEGAL_ARGUMENT_ERROR;
return appendTo;
}
}
CharString buffer;
buffer.appendInvariantChars(count, status);
if (U_FAILURE(status)) {
return appendTo;
}
const SimplePatternFormatter *pattern = getByVariant(buffer.data());
if (pattern == NULL) {
status = U_INVALID_STATE_ERROR;
return appendTo;
}
UnicodeString formattedNumber;
FieldPosition fpos(pos.getField());
if (decFmt != NULL) {
decFmt->format(digits, formattedNumber, fpos, status);
} else {
fmt.format(quantity, formattedNumber, fpos, status);
}
const UnicodeString *params[1] = {&formattedNumber};
int32_t offsets[1];
pattern->formatAndAppend(
params,
UPRV_LENGTHOF(params),
appendTo,
offsets,
UPRV_LENGTHOF(offsets),
status);
if (offsets[0] != -1) {
if (fpos.getBeginIndex() != 0 || fpos.getEndIndex() != 0) {
pos.setBeginIndex(fpos.getBeginIndex() + offsets[0]);
pos.setEndIndex(fpos.getEndIndex() + offsets[0]);
}
}
return appendTo;
}
U_NAMESPACE_END
#endif /* #if !UCONFIG_NO_FORMATTING */