2017-09-27 00:25:20 +00:00
|
|
|
// © 2017 and later: Unicode, Inc. and others.
|
|
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
|
|
|
2017-10-04 22:51:06 +00:00
|
|
|
#include "unicode/utypes.h"
|
|
|
|
|
2017-10-05 00:47:38 +00:00
|
|
|
#if !UCONFIG_NO_FORMATTING && !UPRV_INCOMPLETE_CPP11_SUPPORT
|
2017-09-27 05:31:57 +00:00
|
|
|
|
2017-09-27 00:25:20 +00:00
|
|
|
#include "unicode/numberformatter.h"
|
|
|
|
#include "number_types.h"
|
|
|
|
#include "number_stringbuilder.h"
|
2018-03-15 07:46:56 +00:00
|
|
|
#include "number_decimfmtprops.h"
|
2017-09-27 00:25:20 +00:00
|
|
|
|
2017-10-25 00:25:04 +00:00
|
|
|
using namespace icu;
|
|
|
|
using namespace icu::number;
|
2017-09-27 00:25:20 +00:00
|
|
|
using namespace icu::number::impl;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2018-03-15 07:46:56 +00:00
|
|
|
static UChar32 kFallbackPadChar = 0x0020;
|
|
|
|
|
2017-09-27 00:25:20 +00:00
|
|
|
int32_t
|
|
|
|
addPaddingHelper(UChar32 paddingCp, int32_t requiredPadding, NumberStringBuilder &string, int32_t index,
|
|
|
|
UErrorCode &status) {
|
|
|
|
for (int32_t i = 0; i < requiredPadding; i++) {
|
|
|
|
// TODO: If appending to the end, this will cause actual insertion operations. Improve.
|
|
|
|
string.insertCodePoint(index, paddingCp, UNUM_FIELD_COUNT, status);
|
|
|
|
}
|
|
|
|
return U16_LENGTH(paddingCp) * requiredPadding;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Padder::Padder(UChar32 cp, int32_t width, UNumberFormatPadPosition position) : fWidth(width) {
|
|
|
|
fUnion.padding.fCp = cp;
|
|
|
|
fUnion.padding.fPosition = position;
|
|
|
|
}
|
|
|
|
|
|
|
|
Padder::Padder(int32_t width) : fWidth(width) {}
|
|
|
|
|
|
|
|
Padder Padder::none() {
|
|
|
|
return {-1};
|
|
|
|
}
|
|
|
|
|
|
|
|
Padder Padder::codePoints(UChar32 cp, int32_t targetWidth, UNumberFormatPadPosition position) {
|
|
|
|
// TODO: Validate the code point?
|
|
|
|
if (targetWidth >= 0) {
|
|
|
|
return {cp, targetWidth, position};
|
|
|
|
} else {
|
2018-02-08 06:06:08 +00:00
|
|
|
return {U_NUMBER_ARG_OUTOFBOUNDS_ERROR};
|
2017-09-27 00:25:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-15 07:46:56 +00:00
|
|
|
Padder Padder::forProperties(const DecimalFormatProperties& properties) {
|
|
|
|
UChar32 padCp;
|
|
|
|
if (properties.padString.length() > 0) {
|
|
|
|
padCp = properties.padString.char32At(0);
|
|
|
|
} else {
|
|
|
|
padCp = kFallbackPadChar;
|
|
|
|
}
|
|
|
|
return {padCp, properties.formatWidth, properties.padPosition.getOrDefault(UNUM_PAD_BEFORE_PREFIX)};
|
|
|
|
}
|
|
|
|
|
2017-09-27 06:44:13 +00:00
|
|
|
int32_t Padder::padAndApply(const Modifier &mod1, const Modifier &mod2,
|
|
|
|
NumberStringBuilder &string, int32_t leftIndex, int32_t rightIndex,
|
2017-09-27 00:25:20 +00:00
|
|
|
UErrorCode &status) const {
|
|
|
|
int32_t modLength = mod1.getCodePointCount(status) + mod2.getCodePointCount(status);
|
|
|
|
int32_t requiredPadding = fWidth - modLength - string.codePointCount();
|
|
|
|
U_ASSERT(leftIndex == 0 &&
|
|
|
|
rightIndex == string.length()); // fix the previous line to remove this assertion
|
|
|
|
|
|
|
|
int length = 0;
|
|
|
|
if (requiredPadding <= 0) {
|
|
|
|
// Padding is not required.
|
|
|
|
length += mod1.apply(string, leftIndex, rightIndex, status);
|
|
|
|
length += mod2.apply(string, leftIndex, rightIndex + length, status);
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
PadPosition position = fUnion.padding.fPosition;
|
|
|
|
UChar32 paddingCp = fUnion.padding.fCp;
|
|
|
|
if (position == UNUM_PAD_AFTER_PREFIX) {
|
|
|
|
length += addPaddingHelper(paddingCp, requiredPadding, string, leftIndex, status);
|
|
|
|
} else if (position == UNUM_PAD_BEFORE_SUFFIX) {
|
|
|
|
length += addPaddingHelper(paddingCp, requiredPadding, string, rightIndex + length, status);
|
|
|
|
}
|
|
|
|
length += mod1.apply(string, leftIndex, rightIndex + length, status);
|
|
|
|
length += mod2.apply(string, leftIndex, rightIndex + length, status);
|
|
|
|
if (position == UNUM_PAD_BEFORE_PREFIX) {
|
|
|
|
length += addPaddingHelper(paddingCp, requiredPadding, string, leftIndex, status);
|
|
|
|
} else if (position == UNUM_PAD_AFTER_SUFFIX) {
|
|
|
|
length += addPaddingHelper(paddingCp, requiredPadding, string, rightIndex + length, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
2017-09-27 05:31:57 +00:00
|
|
|
|
|
|
|
#endif /* #if !UCONFIG_NO_FORMATTING */
|