blob: 26fdfafeea4ec217f2afd9c42a16a44aef6ffefc [file] [log] [blame]
Frank Tangf90543d2020-10-30 19:02:04 -07001// © 2020 and later: Unicode, Inc. and others.
2// License & terms of use: http://www.unicode.org/copyright.html
3
4#include "unicode/utypes.h"
5
6#if !UCONFIG_NO_FORMATTING
7
8#include "number_usageprefs.h"
9#include "cstring.h"
10#include "number_decimalquantity.h"
11#include "number_microprops.h"
12#include "number_roundingutils.h"
13#include "number_skeletons.h"
14#include "unicode/char16ptr.h"
15#include "unicode/currunit.h"
16#include "unicode/fmtable.h"
17#include "unicode/measure.h"
18#include "unicode/numberformatter.h"
19#include "unicode/platform.h"
20#include "unicode/unum.h"
21#include "unicode/urename.h"
22#include "units_data.h"
23
24using namespace icu;
25using namespace icu::number;
26using namespace icu::number::impl;
27using icu::StringSegment;
28using icu::units::ConversionRates;
29
30// Copy constructor
Frank Tang7e7574b2021-04-13 21:19:13 -070031StringProp::StringProp(const StringProp &other) : StringProp() {
Frank Tangf90543d2020-10-30 19:02:04 -070032 this->operator=(other);
33}
34
35// Copy assignment operator
Frank Tang7e7574b2021-04-13 21:19:13 -070036StringProp &StringProp::operator=(const StringProp &other) {
37 if (this == &other) { return *this; } // self-assignment: no-op
Frank Tangf90543d2020-10-30 19:02:04 -070038 fLength = 0;
39 fError = other.fError;
Frank Tang7e7574b2021-04-13 21:19:13 -070040 if (fValue != nullptr) {
41 uprv_free(fValue);
42 fValue = nullptr;
Frank Tangf90543d2020-10-30 19:02:04 -070043 }
Frank Tang7e7574b2021-04-13 21:19:13 -070044 if (other.fValue == nullptr) {
Frank Tangf90543d2020-10-30 19:02:04 -070045 return *this;
46 }
47 if (U_FAILURE(other.fError)) {
48 // We don't bother trying to allocating memory if we're in any case busy
Frank Tang7e7574b2021-04-13 21:19:13 -070049 // copying an errored StringProp.
Frank Tangf90543d2020-10-30 19:02:04 -070050 return *this;
51 }
Frank Tang7e7574b2021-04-13 21:19:13 -070052 fValue = (char *)uprv_malloc(other.fLength + 1);
53 if (fValue == nullptr) {
Frank Tangf90543d2020-10-30 19:02:04 -070054 fError = U_MEMORY_ALLOCATION_ERROR;
55 return *this;
56 }
57 fLength = other.fLength;
Frank Tang7e7574b2021-04-13 21:19:13 -070058 uprv_strncpy(fValue, other.fValue, fLength + 1);
Frank Tangf90543d2020-10-30 19:02:04 -070059 return *this;
60}
61
62// Move constructor
Frank Tang7e7574b2021-04-13 21:19:13 -070063StringProp::StringProp(StringProp &&src) U_NOEXCEPT : fValue(src.fValue),
64 fLength(src.fLength),
65 fError(src.fError) {
Frank Tangf90543d2020-10-30 19:02:04 -070066 // Take ownership away from src if necessary
Frank Tang7e7574b2021-04-13 21:19:13 -070067 src.fValue = nullptr;
Frank Tangf90543d2020-10-30 19:02:04 -070068}
69
70// Move assignment operator
Frank Tang7e7574b2021-04-13 21:19:13 -070071StringProp &StringProp::operator=(StringProp &&src) U_NOEXCEPT {
Frank Tangf90543d2020-10-30 19:02:04 -070072 if (this == &src) {
73 return *this;
74 }
Frank Tang7e7574b2021-04-13 21:19:13 -070075 if (fValue != nullptr) {
76 uprv_free(fValue);
Frank Tangf90543d2020-10-30 19:02:04 -070077 }
Frank Tang7e7574b2021-04-13 21:19:13 -070078 fValue = src.fValue;
Frank Tangf90543d2020-10-30 19:02:04 -070079 fLength = src.fLength;
80 fError = src.fError;
81 // Take ownership away from src if necessary
Frank Tang7e7574b2021-04-13 21:19:13 -070082 src.fValue = nullptr;
Frank Tangf90543d2020-10-30 19:02:04 -070083 return *this;
84}
85
Frank Tang7e7574b2021-04-13 21:19:13 -070086StringProp::~StringProp() {
87 if (fValue != nullptr) {
88 uprv_free(fValue);
89 fValue = nullptr;
Frank Tangf90543d2020-10-30 19:02:04 -070090 }
91}
92
Frank Tang7e7574b2021-04-13 21:19:13 -070093void StringProp::set(StringPiece value) {
94 if (fValue != nullptr) {
95 uprv_free(fValue);
96 fValue = nullptr;
Frank Tangf90543d2020-10-30 19:02:04 -070097 }
98 fLength = value.length();
Frank Tang7e7574b2021-04-13 21:19:13 -070099 fValue = (char *)uprv_malloc(fLength + 1);
100 if (fValue == nullptr) {
Frank Tangf90543d2020-10-30 19:02:04 -0700101 fLength = 0;
102 fError = U_MEMORY_ALLOCATION_ERROR;
103 return;
104 }
Frank Tang7e7574b2021-04-13 21:19:13 -0700105 uprv_strncpy(fValue, value.data(), fLength);
106 fValue[fLength] = 0;
Frank Tangf90543d2020-10-30 19:02:04 -0700107}
108
109// Populates micros.mixedMeasures and modifies quantity, based on the values in
110// measures.
111void mixedMeasuresToMicros(const MaybeStackVector<Measure> &measures, DecimalQuantity *quantity,
112 MicroProps *micros, UErrorCode status) {
Frank Tang7e7574b2021-04-13 21:19:13 -0700113 micros->mixedMeasuresCount = measures.length();
114
115 if (micros->mixedMeasures.getCapacity() < micros->mixedMeasuresCount) {
116 if (micros->mixedMeasures.resize(micros->mixedMeasuresCount) == nullptr) {
117 status = U_MEMORY_ALLOCATION_ERROR;
118 return;
Frank Tangf90543d2020-10-30 19:02:04 -0700119 }
Frank Tangf90543d2020-10-30 19:02:04 -0700120 }
Frank Tang7e7574b2021-04-13 21:19:13 -0700121
122 for (int32_t i = 0; i < micros->mixedMeasuresCount; i++) {
123 switch (measures[i]->getNumber().getType()) {
124 case Formattable::kInt64:
125 micros->mixedMeasures[i] = measures[i]->getNumber().getInt64();
126 break;
127
128 case Formattable::kDouble:
129 U_ASSERT(micros->indexOfQuantity < 0);
130 quantity->setToDouble(measures[i]->getNumber().getDouble());
131 micros->indexOfQuantity = i;
132 break;
133
134 default:
135 U_ASSERT(0 == "Found a Measure Number which is neither a double nor an int");
Frank Tang3e05d9d2021-11-08 14:04:04 -0800136 UPRV_UNREACHABLE_EXIT;
Frank Tang7e7574b2021-04-13 21:19:13 -0700137 break;
138 }
139
140 if (U_FAILURE(status)) {
141 return;
142 }
143 }
144
145 if (micros->indexOfQuantity < 0) {
146 // There is no quantity.
147 status = U_INTERNAL_PROGRAM_ERROR;
148 }
Frank Tangf90543d2020-10-30 19:02:04 -0700149}
150
151UsagePrefsHandler::UsagePrefsHandler(const Locale &locale,
152 const MeasureUnit &inputUnit,
153 const StringPiece usage,
154 const MicroPropsGenerator *parent,
155 UErrorCode &status)
Frank Tang1f164ee2022-11-08 12:31:27 -0800156 : fUnitsRouter(inputUnit, locale, usage, status),
Frank Tangf90543d2020-10-30 19:02:04 -0700157 fParent(parent) {
158}
159
160void UsagePrefsHandler::processQuantity(DecimalQuantity &quantity, MicroProps &micros,
161 UErrorCode &status) const {
162 fParent->processQuantity(quantity, micros, status);
163 if (U_FAILURE(status)) {
164 return;
165 }
166
167 quantity.roundToInfinity(); // Enables toDouble
168 const units::RouteResult routed = fUnitsRouter.route(quantity.toDouble(), &micros.rounder, status);
169 if (U_FAILURE(status)) {
170 return;
171 }
172 const MaybeStackVector<Measure>& routedMeasures = routed.measures;
173 micros.outputUnit = routed.outputUnit.copy(status).build(status);
174 if (U_FAILURE(status)) {
175 return;
176 }
177
178 mixedMeasuresToMicros(routedMeasures, &quantity, &micros, status);
179}
180
Frank Tang7e7574b2021-04-13 21:19:13 -0700181UnitConversionHandler::UnitConversionHandler(const MeasureUnit &targetUnit,
Frank Tangf90543d2020-10-30 19:02:04 -0700182 const MicroPropsGenerator *parent, UErrorCode &status)
Frank Tang7e7574b2021-04-13 21:19:13 -0700183 : fOutputUnit(targetUnit), fParent(parent) {
Frank Tangf90543d2020-10-30 19:02:04 -0700184 MeasureUnitImpl tempInput, tempOutput;
Frank Tangf90543d2020-10-30 19:02:04 -0700185
Frank Tangf90543d2020-10-30 19:02:04 -0700186 ConversionRates conversionRates(status);
187 if (U_FAILURE(status)) {
188 return;
189 }
Frank Tang7e7574b2021-04-13 21:19:13 -0700190
191 const MeasureUnitImpl &targetUnitImpl =
192 MeasureUnitImpl::forMeasureUnit(targetUnit, tempOutput, status);
Frank Tangf90543d2020-10-30 19:02:04 -0700193 fUnitConverter.adoptInsteadAndCheckErrorCode(
Frank Tang7e7574b2021-04-13 21:19:13 -0700194 new ComplexUnitsConverter(targetUnitImpl, conversionRates, status), status);
Frank Tangf90543d2020-10-30 19:02:04 -0700195}
196
197void UnitConversionHandler::processQuantity(DecimalQuantity &quantity, MicroProps &micros,
198 UErrorCode &status) const {
199 fParent->processQuantity(quantity, micros, status);
200 if (U_FAILURE(status)) {
201 return;
202 }
203 quantity.roundToInfinity(); // Enables toDouble
204 MaybeStackVector<Measure> measures =
205 fUnitConverter->convert(quantity.toDouble(), &micros.rounder, status);
206 micros.outputUnit = fOutputUnit;
207 if (U_FAILURE(status)) {
208 return;
209 }
210
211 mixedMeasuresToMicros(measures, &quantity, &micros, status);
212}
213
214#endif /* #if !UCONFIG_NO_FORMATTING */