1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
|
// © 2018 and later: Unicode, Inc. and others.
// License & terms of use: http://www.unicode.org/copyright.html
#include <_foundation_unicode/utypes.h>
#if !UCONFIG_NO_FORMATTING
// Allow implicit conversion from char16_t* to UnicodeString for this file:
// Helpful in toString methods and elsewhere.
#define UNISTR_FROM_STRING_EXPLICIT
#include "numparse_types.h"
#include "number_decimalquantity.h"
#include "string_segment.h"
#include "putilimp.h"
#include <cmath>
using namespace icu;
using namespace icu::number;
using namespace icu::number::impl;
using namespace icu::numparse;
using namespace icu::numparse::impl;
ParsedNumber::ParsedNumber() {
clear();
}
void ParsedNumber::clear() {
quantity.bogus = true;
charEnd = 0;
flags = 0;
prefix.setToBogus();
suffix.setToBogus();
currencyCode[0] = 0;
}
void ParsedNumber::setCharsConsumed(const StringSegment& segment) {
charEnd = segment.getOffset();
}
void ParsedNumber::postProcess() {
if (!quantity.bogus && 0 != (flags & FLAG_NEGATIVE)) {
quantity.negate();
}
}
bool ParsedNumber::success() const {
return charEnd > 0 && 0 == (flags & FLAG_FAIL);
}
bool ParsedNumber::seenNumber() const {
return !quantity.bogus || 0 != (flags & FLAG_NAN) || 0 != (flags & FLAG_INFINITY);
}
double ParsedNumber::getDouble(UErrorCode& status) const {
bool sawNaN = 0 != (flags & FLAG_NAN);
bool sawInfinity = 0 != (flags & FLAG_INFINITY);
// Check for NaN, infinity, and -0.0
if (sawNaN) {
// Can't use NAN or std::nan because the byte pattern is platform-dependent;
// MSVC sets the sign bit, but Clang and GCC do not
return uprv_getNaN();
}
if (sawInfinity) {
if (0 != (flags & FLAG_NEGATIVE)) {
return -INFINITY;
} else {
return INFINITY;
}
}
if (quantity.bogus) {
status = U_INVALID_STATE_ERROR;
return 0.0;
}
if (quantity.isZeroish() && quantity.isNegative()) {
return -0.0;
}
if (quantity.fitsInLong()) {
return static_cast<double>(quantity.toLong());
} else {
return quantity.toDouble();
}
}
void ParsedNumber::populateFormattable(Formattable& output, parse_flags_t parseFlags) const {
bool sawNaN = 0 != (flags & FLAG_NAN);
bool sawInfinity = 0 != (flags & FLAG_INFINITY);
bool integerOnly = 0 != (parseFlags & PARSE_FLAG_INTEGER_ONLY);
// Check for NaN, infinity, and -0.0
if (sawNaN) {
// Can't use NAN or std::nan because the byte pattern is platform-dependent;
// MSVC sets the sign bit, but Clang and GCC do not
output.setDouble(uprv_getNaN());
return;
}
if (sawInfinity) {
if (0 != (flags & FLAG_NEGATIVE)) {
output.setDouble(-INFINITY);
return;
} else {
output.setDouble(INFINITY);
return;
}
}
U_ASSERT(!quantity.bogus);
if (quantity.isZeroish() && quantity.isNegative() && !integerOnly) {
output.setDouble(-0.0);
return;
}
// All other numbers
output.adoptDecimalQuantity(new DecimalQuantity(quantity));
}
bool ParsedNumber::isBetterThan(const ParsedNumber& other) {
// Favor results with strictly more characters consumed.
return charEnd > other.charEnd;
}
#endif /* #if !UCONFIG_NO_FORMATTING */
|