aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/icu/i18n/numparse_parsednumber.cpp
blob: cd2dad8883e3c5af80fb547fca198de04419e45d (plain) (blame)
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 "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 */