]> git.saurik.com Git - apple/icu.git/blame - icuSources/i18n/nfsubs.cpp
ICU-59180.0.1.tar.gz
[apple/icu.git] / icuSources / i18n / nfsubs.cpp
CommitLineData
f3c0d7a5
A
1// © 2016 and later: Unicode, Inc. and others.
2// License & terms of use: http://www.unicode.org/copyright.html
b75a7d8f
A
3/*
4******************************************************************************
2ca993e8 5* Copyright (C) 1997-2015, International Business Machines
b75a7d8f
A
6* Corporation and others. All Rights Reserved.
7******************************************************************************
8* file name: nfsubs.cpp
f3c0d7a5 9* encoding: UTF-8
b75a7d8f
A
10* tab size: 8 (not used)
11* indentation:4
12*
13* Modification history
14* Date Name Comments
15* 10/11/2001 Doug Ported from ICU4J
16*/
17
729e4ab9 18#include <stdio.h>
51004dcb 19#include "utypeinfo.h" // for 'typeid' to work
729e4ab9 20
b75a7d8f 21#include "nfsubs.h"
374ca955 22#include "digitlst.h"
f3c0d7a5 23#include "fmtableimp.h"
374ca955 24
b75a7d8f
A
25#if U_HAVE_RBNF
26
27static const UChar gLessThan = 0x003c;
28static const UChar gEquals = 0x003d;
29static const UChar gGreaterThan = 0x003e;
30static const UChar gPercent = 0x0025;
31static const UChar gPound = 0x0023;
32static const UChar gZero = 0x0030;
33static const UChar gSpace = 0x0020;
34
35static const UChar gEqualsEquals[] =
36{
37 0x3D, 0x3D, 0
38}; /* "==" */
39static const UChar gGreaterGreaterGreaterThan[] =
40{
41 0x3E, 0x3E, 0x3E, 0
42}; /* ">>>" */
43static const UChar gGreaterGreaterThan[] =
44{
45 0x3E, 0x3E, 0
46}; /* ">>" */
47
46f4442e
A
48U_NAMESPACE_BEGIN
49
50class SameValueSubstitution : public NFSubstitution {
51public:
52 SameValueSubstitution(int32_t pos,
53 const NFRuleSet* ruleset,
46f4442e
A
54 const UnicodeString& description,
55 UErrorCode& status);
4388f060 56 virtual ~SameValueSubstitution();
46f4442e
A
57
58 virtual int64_t transformNumber(int64_t number) const { return number; }
59 virtual double transformNumber(double number) const { return number; }
60 virtual double composeRuleValue(double newRuleValue, double /*oldRuleValue*/) const { return newRuleValue; }
61 virtual double calcUpperBound(double oldUpperBound) const { return oldUpperBound; }
62 virtual UChar tokenChar() const { return (UChar)0x003d; } // '='
63
64public:
65 static UClassID getStaticClassID(void);
66 virtual UClassID getDynamicClassID(void) const;
67};
68
4388f060
A
69SameValueSubstitution::~SameValueSubstitution() {}
70
46f4442e 71class MultiplierSubstitution : public NFSubstitution {
f3c0d7a5 72 int64_t divisor;
46f4442e
A
73
74public:
75 MultiplierSubstitution(int32_t _pos,
f3c0d7a5 76 const NFRule *rule,
46f4442e 77 const NFRuleSet* _ruleSet,
46f4442e
A
78 const UnicodeString& description,
79 UErrorCode& status)
f3c0d7a5 80 : NFSubstitution(_pos, _ruleSet, description, status), divisor(rule->getDivisor())
46f4442e 81 {
46f4442e
A
82 if (divisor == 0) {
83 status = U_PARSE_ERROR;
84 }
85 }
4388f060 86 virtual ~MultiplierSubstitution();
46f4442e 87
f3c0d7a5
A
88 virtual void setDivisor(int32_t radix, int16_t exponent, UErrorCode& status) {
89 divisor = util64_pow(radix, exponent);
46f4442e
A
90
91 if(divisor == 0) {
92 status = U_PARSE_ERROR;
93 }
94 }
95
96 virtual UBool operator==(const NFSubstitution& rhs) const;
97
98 virtual int64_t transformNumber(int64_t number) const {
f3c0d7a5 99 return number / divisor;
46f4442e
A
100 }
101
102 virtual double transformNumber(double number) const {
103 if (getRuleSet()) {
104 return uprv_floor(number / divisor);
105 } else {
f3c0d7a5 106 return number / divisor;
46f4442e
A
107 }
108 }
109
110 virtual double composeRuleValue(double newRuleValue, double /*oldRuleValue*/) const {
111 return newRuleValue * divisor;
112 }
113
114 virtual double calcUpperBound(double /*oldUpperBound*/) const { return divisor; }
115
116 virtual UChar tokenChar() const { return (UChar)0x003c; } // '<'
117
118public:
119 static UClassID getStaticClassID(void);
120 virtual UClassID getDynamicClassID(void) const;
121};
122
4388f060
A
123MultiplierSubstitution::~MultiplierSubstitution() {}
124
46f4442e 125class ModulusSubstitution : public NFSubstitution {
f3c0d7a5 126 int64_t divisor;
46f4442e
A
127 const NFRule* ruleToUse;
128public:
129 ModulusSubstitution(int32_t pos,
f3c0d7a5 130 const NFRule* rule,
46f4442e
A
131 const NFRule* rulePredecessor,
132 const NFRuleSet* ruleSet,
46f4442e
A
133 const UnicodeString& description,
134 UErrorCode& status);
4388f060 135 virtual ~ModulusSubstitution();
46f4442e 136
f3c0d7a5
A
137 virtual void setDivisor(int32_t radix, int16_t exponent, UErrorCode& status) {
138 divisor = util64_pow(radix, exponent);
46f4442e
A
139
140 if (divisor == 0) {
141 status = U_PARSE_ERROR;
142 }
143 }
144
145 virtual UBool operator==(const NFSubstitution& rhs) const;
146
2ca993e8
A
147 virtual void doSubstitution(int64_t number, UnicodeString& toInsertInto, int32_t pos, int32_t recursionCount, UErrorCode& status) const;
148 virtual void doSubstitution(double number, UnicodeString& toInsertInto, int32_t pos, int32_t recursionCount, UErrorCode& status) const;
46f4442e 149
f3c0d7a5 150 virtual int64_t transformNumber(int64_t number) const { return number % divisor; }
46f4442e
A
151 virtual double transformNumber(double number) const { return uprv_fmod(number, divisor); }
152
153 virtual UBool doParse(const UnicodeString& text,
154 ParsePosition& parsePosition,
155 double baseValue,
156 double upperBound,
157 UBool lenientParse,
158 Formattable& result) const;
159
160 virtual double composeRuleValue(double newRuleValue, double oldRuleValue) const {
161 return oldRuleValue - uprv_fmod(oldRuleValue, divisor) + newRuleValue;
162 }
163
164 virtual double calcUpperBound(double /*oldUpperBound*/) const { return divisor; }
165
166 virtual UBool isModulusSubstitution() const { return TRUE; }
167
168 virtual UChar tokenChar() const { return (UChar)0x003e; } // '>'
169
4388f060
A
170 virtual void toString(UnicodeString& result) const;
171
46f4442e
A
172public:
173 static UClassID getStaticClassID(void);
174 virtual UClassID getDynamicClassID(void) const;
175};
176
4388f060
A
177ModulusSubstitution::~ModulusSubstitution() {}
178
46f4442e
A
179class IntegralPartSubstitution : public NFSubstitution {
180public:
181 IntegralPartSubstitution(int32_t _pos,
182 const NFRuleSet* _ruleSet,
46f4442e
A
183 const UnicodeString& description,
184 UErrorCode& status)
2ca993e8 185 : NFSubstitution(_pos, _ruleSet, description, status) {}
4388f060 186 virtual ~IntegralPartSubstitution();
46f4442e
A
187
188 virtual int64_t transformNumber(int64_t number) const { return number; }
189 virtual double transformNumber(double number) const { return uprv_floor(number); }
190 virtual double composeRuleValue(double newRuleValue, double oldRuleValue) const { return newRuleValue + oldRuleValue; }
191 virtual double calcUpperBound(double /*oldUpperBound*/) const { return DBL_MAX; }
192 virtual UChar tokenChar() const { return (UChar)0x003c; } // '<'
193
194public:
195 static UClassID getStaticClassID(void);
196 virtual UClassID getDynamicClassID(void) const;
197};
198
4388f060
A
199IntegralPartSubstitution::~IntegralPartSubstitution() {}
200
46f4442e
A
201class FractionalPartSubstitution : public NFSubstitution {
202 UBool byDigits;
203 UBool useSpaces;
204 enum { kMaxDecimalDigits = 8 };
205public:
206 FractionalPartSubstitution(int32_t pos,
207 const NFRuleSet* ruleSet,
46f4442e
A
208 const UnicodeString& description,
209 UErrorCode& status);
4388f060 210 virtual ~FractionalPartSubstitution();
46f4442e
A
211
212 virtual UBool operator==(const NFSubstitution& rhs) const;
213
2ca993e8
A
214 virtual void doSubstitution(double number, UnicodeString& toInsertInto, int32_t pos, int32_t recursionCount, UErrorCode& status) const;
215 virtual void doSubstitution(int64_t /*number*/, UnicodeString& /*toInsertInto*/, int32_t /*_pos*/, int32_t /*recursionCount*/, UErrorCode& /*status*/) const {}
46f4442e
A
216 virtual int64_t transformNumber(int64_t /*number*/) const { return 0; }
217 virtual double transformNumber(double number) const { return number - uprv_floor(number); }
218
219 virtual UBool doParse(const UnicodeString& text,
220 ParsePosition& parsePosition,
221 double baseValue,
222 double upperBound,
223 UBool lenientParse,
224 Formattable& result) const;
225
226 virtual double composeRuleValue(double newRuleValue, double oldRuleValue) const { return newRuleValue + oldRuleValue; }
227 virtual double calcUpperBound(double /*oldUpperBound*/) const { return 0.0; }
228 virtual UChar tokenChar() const { return (UChar)0x003e; } // '>'
229
230public:
231 static UClassID getStaticClassID(void);
232 virtual UClassID getDynamicClassID(void) const;
233};
234
4388f060
A
235FractionalPartSubstitution::~FractionalPartSubstitution() {}
236
46f4442e
A
237class AbsoluteValueSubstitution : public NFSubstitution {
238public:
239 AbsoluteValueSubstitution(int32_t _pos,
240 const NFRuleSet* _ruleSet,
46f4442e
A
241 const UnicodeString& description,
242 UErrorCode& status)
2ca993e8 243 : NFSubstitution(_pos, _ruleSet, description, status) {}
4388f060 244 virtual ~AbsoluteValueSubstitution();
46f4442e
A
245
246 virtual int64_t transformNumber(int64_t number) const { return number >= 0 ? number : -number; }
247 virtual double transformNumber(double number) const { return uprv_fabs(number); }
248 virtual double composeRuleValue(double newRuleValue, double /*oldRuleValue*/) const { return -newRuleValue; }
249 virtual double calcUpperBound(double /*oldUpperBound*/) const { return DBL_MAX; }
250 virtual UChar tokenChar() const { return (UChar)0x003e; } // '>'
251
252public:
253 static UClassID getStaticClassID(void);
254 virtual UClassID getDynamicClassID(void) const;
255};
256
4388f060
A
257AbsoluteValueSubstitution::~AbsoluteValueSubstitution() {}
258
46f4442e
A
259class NumeratorSubstitution : public NFSubstitution {
260 double denominator;
261 int64_t ldenominator;
262 UBool withZeros;
263public:
264 static inline UnicodeString fixdesc(const UnicodeString& desc) {
265 if (desc.endsWith(LTLT, 2)) {
266 UnicodeString result(desc, 0, desc.length()-1);
267 return result;
268 }
269 return desc;
270 }
271 NumeratorSubstitution(int32_t _pos,
272 double _denominator,
2ca993e8 273 NFRuleSet* _ruleSet,
46f4442e
A
274 const UnicodeString& description,
275 UErrorCode& status)
2ca993e8 276 : NFSubstitution(_pos, _ruleSet, fixdesc(description), status), denominator(_denominator)
46f4442e
A
277 {
278 ldenominator = util64_fromDouble(denominator);
279 withZeros = description.endsWith(LTLT, 2);
280 }
4388f060 281 virtual ~NumeratorSubstitution();
46f4442e
A
282
283 virtual UBool operator==(const NFSubstitution& rhs) const;
284
285 virtual int64_t transformNumber(int64_t number) const { return number * ldenominator; }
286 virtual double transformNumber(double number) const { return uprv_round(number * denominator); }
287
2ca993e8
A
288 virtual void doSubstitution(int64_t /*number*/, UnicodeString& /*toInsertInto*/, int32_t /*_pos*/, int32_t /*recursionCount*/, UErrorCode& /*status*/) const {}
289 virtual void doSubstitution(double number, UnicodeString& toInsertInto, int32_t pos, int32_t recursionCount, UErrorCode& status) const;
46f4442e
A
290 virtual UBool doParse(const UnicodeString& text,
291 ParsePosition& parsePosition,
292 double baseValue,
293 double upperBound,
294 UBool /*lenientParse*/,
295 Formattable& result) const;
296
297 virtual double composeRuleValue(double newRuleValue, double oldRuleValue) const { return newRuleValue / oldRuleValue; }
298 virtual double calcUpperBound(double /*oldUpperBound*/) const { return denominator; }
299 virtual UChar tokenChar() const { return (UChar)0x003c; } // '<'
300private:
301 static const UChar LTLT[2];
302
303public:
304 static UClassID getStaticClassID(void);
305 virtual UClassID getDynamicClassID(void) const;
306};
307
4388f060
A
308NumeratorSubstitution::~NumeratorSubstitution() {}
309
b75a7d8f
A
310NFSubstitution*
311NFSubstitution::makeSubstitution(int32_t pos,
312 const NFRule* rule,
313 const NFRule* predecessor,
314 const NFRuleSet* ruleSet,
315 const RuleBasedNumberFormat* formatter,
316 const UnicodeString& description,
317 UErrorCode& status)
318{
319 // if the description is empty, return a NullSubstitution
320 if (description.length() == 0) {
2ca993e8 321 return NULL;
b75a7d8f
A
322 }
323
324 switch (description.charAt(0)) {
325 // if the description begins with '<'...
326 case gLessThan:
327 // throw an exception if the rule is a negative number
328 // rule
329 if (rule->getBaseValue() == NFRule::kNegativeNumberRule) {
330 // throw new IllegalArgumentException("<< not allowed in negative-number rule");
331 status = U_PARSE_ERROR;
332 return NULL;
333 }
334
335 // if the rule is a fraction rule, return an
336 // IntegralPartSubstitution
337 else if (rule->getBaseValue() == NFRule::kImproperFractionRule
338 || rule->getBaseValue() == NFRule::kProperFractionRule
339 || rule->getBaseValue() == NFRule::kMasterRule) {
2ca993e8 340 return new IntegralPartSubstitution(pos, ruleSet, description, status);
b75a7d8f
A
341 }
342
343 // if the rule set containing the rule is a fraction
344 // rule set, return a NumeratorSubstitution
345 else if (ruleSet->isFractionRuleSet()) {
346 return new NumeratorSubstitution(pos, (double)rule->getBaseValue(),
2ca993e8 347 formatter->getDefaultRuleSet(), description, status);
b75a7d8f
A
348 }
349
350 // otherwise, return a MultiplierSubstitution
351 else {
f3c0d7a5 352 return new MultiplierSubstitution(pos, rule, ruleSet,
2ca993e8 353 description, status);
b75a7d8f
A
354 }
355
356 // if the description begins with '>'...
357 case gGreaterThan:
358 // if the rule is a negative-number rule, return
359 // an AbsoluteValueSubstitution
360 if (rule->getBaseValue() == NFRule::kNegativeNumberRule) {
2ca993e8 361 return new AbsoluteValueSubstitution(pos, ruleSet, description, status);
b75a7d8f
A
362 }
363
364 // if the rule is a fraction rule, return a
365 // FractionalPartSubstitution
366 else if (rule->getBaseValue() == NFRule::kImproperFractionRule
367 || rule->getBaseValue() == NFRule::kProperFractionRule
368 || rule->getBaseValue() == NFRule::kMasterRule) {
2ca993e8 369 return new FractionalPartSubstitution(pos, ruleSet, description, status);
b75a7d8f
A
370 }
371
372 // if the rule set owning the rule is a fraction rule set,
373 // throw an exception
374 else if (ruleSet->isFractionRuleSet()) {
375 // throw new IllegalArgumentException(">> not allowed in fraction rule set");
376 status = U_PARSE_ERROR;
377 return NULL;
378 }
379
380 // otherwise, return a ModulusSubstitution
381 else {
f3c0d7a5 382 return new ModulusSubstitution(pos, rule, predecessor,
2ca993e8 383 ruleSet, description, status);
b75a7d8f
A
384 }
385
386 // if the description begins with '=', always return a
387 // SameValueSubstitution
388 case gEquals:
2ca993e8 389 return new SameValueSubstitution(pos, ruleSet, description, status);
b75a7d8f
A
390
391 // and if it's anything else, throw an exception
392 default:
393 // throw new IllegalArgumentException("Illegal substitution character");
394 status = U_PARSE_ERROR;
395 }
396 return NULL;
397}
398
399NFSubstitution::NFSubstitution(int32_t _pos,
400 const NFRuleSet* _ruleSet,
b75a7d8f
A
401 const UnicodeString& description,
402 UErrorCode& status)
403 : pos(_pos), ruleSet(NULL), numberFormat(NULL)
404{
405 // the description should begin and end with the same character.
406 // If it doesn't that's a syntax error. Otherwise,
407 // makeSubstitution() was the only thing that needed to know
408 // about these characters, so strip them off
409 UnicodeString workingDescription(description);
410 if (description.length() >= 2
411 && description.charAt(0) == description.charAt(description.length() - 1))
412 {
413 workingDescription.remove(description.length() - 1, 1);
414 workingDescription.remove(0, 1);
415 }
416 else if (description.length() != 0) {
417 // throw new IllegalArgumentException("Illegal substitution syntax");
418 status = U_PARSE_ERROR;
419 return;
420 }
421
b75a7d8f 422 if (workingDescription.length() == 0) {
2ca993e8
A
423 // if the description was just two paired token characters
424 // (i.e., "<<" or ">>"), it uses the rule set it belongs to to
425 // format its result
b75a7d8f
A
426 this->ruleSet = _ruleSet;
427 }
b75a7d8f 428 else if (workingDescription.charAt(0) == gPercent) {
2ca993e8
A
429 // if the description contains a rule set name, that's the rule
430 // set we use to format the result: get a reference to the
431 // names rule set
432 this->ruleSet = _ruleSet->getOwner()->findRuleSet(workingDescription, status);
b75a7d8f 433 }
b75a7d8f 434 else if (workingDescription.charAt(0) == gPound || workingDescription.charAt(0) ==gZero) {
2ca993e8
A
435 // if the description begins with 0 or #, treat it as a
436 // DecimalFormat pattern, and initialize a DecimalFormat with
437 // that pattern (then set it to use the DecimalFormatSymbols
438 // belonging to our formatter)
439 const DecimalFormatSymbols* sym = _ruleSet->getOwner()->getDecimalFormatSymbols();
374ca955
A
440 if (!sym) {
441 status = U_MISSING_RESOURCE_ERROR;
442 return;
443 }
2ca993e8 444 DecimalFormat *tempNumberFormat = new DecimalFormat(workingDescription, *sym, status);
b75a7d8f 445 /* test for NULL */
2ca993e8 446 if (!tempNumberFormat) {
b75a7d8f
A
447 status = U_MEMORY_ALLOCATION_ERROR;
448 return;
449 }
374ca955 450 if (U_FAILURE(status)) {
2ca993e8 451 delete tempNumberFormat;
374ca955
A
452 return;
453 }
2ca993e8 454 this->numberFormat = tempNumberFormat;
b75a7d8f 455 }
b75a7d8f 456 else if (workingDescription.charAt(0) == gGreaterThan) {
2ca993e8
A
457 // if the description is ">>>", this substitution bypasses the
458 // usual rule-search process and always uses the rule that precedes
459 // it in its own rule set's rule list (this is used for place-value
460 // notations: formats where you want to see a particular part of
461 // a number even when it's 0)
462
b75a7d8f
A
463 // this causes problems when >>> is used in a frationalPartSubstitution
464 // this->ruleSet = NULL;
465 this->ruleSet = _ruleSet;
466 this->numberFormat = NULL;
467 }
b75a7d8f 468 else {
2ca993e8
A
469 // and of the description is none of these things, it's a syntax error
470
b75a7d8f
A
471 // throw new IllegalArgumentException("Illegal substitution syntax");
472 status = U_PARSE_ERROR;
473 }
474}
475
476NFSubstitution::~NFSubstitution()
477{
2ca993e8
A
478 delete numberFormat;
479 numberFormat = NULL;
b75a7d8f
A
480}
481
482/**
483 * Set's the substitution's divisor. Used by NFRule.setBaseValue().
484 * A no-op for all substitutions except multiplier and modulus
485 * substitutions.
486 * @param radix The radix of the divisor
487 * @param exponent The exponent of the divisor
488 */
489void
f3c0d7a5 490NFSubstitution::setDivisor(int32_t /*radix*/, int16_t /*exponent*/, UErrorCode& /*status*/) {
b75a7d8f
A
491 // a no-op for all substitutions except multiplier and modulus substitutions
492}
493
2ca993e8
A
494void
495NFSubstitution::setDecimalFormatSymbols(const DecimalFormatSymbols &newSymbols, UErrorCode& /*status*/) {
496 if (numberFormat != NULL) {
497 numberFormat->setDecimalFormatSymbols(newSymbols);
498 }
499}
b75a7d8f
A
500
501//-----------------------------------------------------------------------
502// boilerplate
503//-----------------------------------------------------------------------
504
46f4442e 505UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NFSubstitution)
b75a7d8f 506
46f4442e
A
507/**
508 * Compares two substitutions for equality
509 * @param The substitution to compare this one to
510 * @return true if the two substitutions are functionally equivalent
511 */
b75a7d8f
A
512UBool
513NFSubstitution::operator==(const NFSubstitution& rhs) const
514{
515 // compare class and all of the fields all substitutions have
516 // in common
517 // this should be called by subclasses before their own equality tests
729e4ab9 518 return typeid(*this) == typeid(rhs)
b75a7d8f
A
519 && pos == rhs.pos
520 && (ruleSet == NULL) == (rhs.ruleSet == NULL)
521 // && ruleSet == rhs.ruleSet causes circularity, other checks to make instead?
522 && (numberFormat == NULL
523 ? (rhs.numberFormat == NULL)
524 : (*numberFormat == *rhs.numberFormat));
525}
526
46f4442e
A
527/**
528 * Returns a textual description of the substitution
529 * @return A textual description of the substitution. This might
530 * not be identical to the description it was created from, but
531 * it'll produce the same result.
532 */
b75a7d8f
A
533void
534NFSubstitution::toString(UnicodeString& text) const
535{
536 // use tokenChar() to get the character at the beginning and
537 // end of the substitutin token. In between them will go
538 // either the name of the rule set it uses, or the pattern of
539 // the DecimalFormat it uses
540 text.remove();
541 text.append(tokenChar());
542
543 UnicodeString temp;
544 if (ruleSet != NULL) {
545 ruleSet->getName(temp);
73c04bcf 546 } else if (numberFormat != NULL) {
b75a7d8f
A
547 numberFormat->toPattern(temp);
548 }
549 text.append(temp);
550 text.append(tokenChar());
551}
552
553//-----------------------------------------------------------------------
554// formatting
555//-----------------------------------------------------------------------
556
557/**
558 * Performs a mathematical operation on the number, formats it using
559 * either ruleSet or decimalFormat, and inserts the result into
560 * toInsertInto.
561 * @param number The number being formatted.
562 * @param toInsertInto The string we insert the result into
563 * @param pos The position in toInsertInto where the owning rule's
564 * rule text begins (this value is added to this substitution's
565 * position to determine exactly where to insert the new text)
566 */
567void
2ca993e8 568NFSubstitution::doSubstitution(int64_t number, UnicodeString& toInsertInto, int32_t _pos, int32_t recursionCount, UErrorCode& status) const
b75a7d8f
A
569{
570 if (ruleSet != NULL) {
f3c0d7a5 571 // Perform a transformation on the number that is dependent
b75a7d8f
A
572 // on the type of substitution this is, then just call its
573 // rule set's format() method to format the result
2ca993e8 574 ruleSet->format(transformNumber(number), toInsertInto, _pos + this->pos, recursionCount, status);
73c04bcf 575 } else if (numberFormat != NULL) {
f3c0d7a5
A
576 if (number <= MAX_INT64_IN_DOUBLE) {
577 // or perform the transformation on the number (preserving
578 // the result's fractional part if the formatter it set
579 // to show it), then use that formatter's format() method
580 // to format the result
581 double numberToFormat = transformNumber((double)number);
582 if (numberFormat->getMaximumFractionDigits() == 0) {
583 numberToFormat = uprv_floor(numberToFormat);
584 }
b75a7d8f 585
f3c0d7a5
A
586 UnicodeString temp;
587 numberFormat->format(numberToFormat, temp, status);
588 toInsertInto.insert(_pos + this->pos, temp);
589 }
590 else {
591 // We have gone beyond double precision. Something has to give.
592 // We're favoring accuracy of the large number over potential rules
593 // that round like a CompactDecimalFormat, which is not a common use case.
594 //
595 // Perform a transformation on the number that is dependent
596 // on the type of substitution this is, then just call its
597 // rule set's format() method to format the result
598 int64_t numberToFormat = transformNumber(number);
599 UnicodeString temp;
600 numberFormat->format(numberToFormat, temp, status);
601 toInsertInto.insert(_pos + this->pos, temp);
602 }
b75a7d8f
A
603 }
604}
605
606/**
607 * Performs a mathematical operation on the number, formats it using
608 * either ruleSet or decimalFormat, and inserts the result into
609 * toInsertInto.
610 * @param number The number being formatted.
611 * @param toInsertInto The string we insert the result into
612 * @param pos The position in toInsertInto where the owning rule's
613 * rule text begins (this value is added to this substitution's
614 * position to determine exactly where to insert the new text)
615 */
616void
2ca993e8 617NFSubstitution::doSubstitution(double number, UnicodeString& toInsertInto, int32_t _pos, int32_t recursionCount, UErrorCode& status) const {
b75a7d8f
A
618 // perform a transformation on the number being formatted that
619 // is dependent on the type of substitution this is
620 double numberToFormat = transformNumber(number);
2ca993e8
A
621
622 if (uprv_isInfinite(numberToFormat)) {
623 // This is probably a minus rule. Combine it with an infinite rule.
624 const NFRule *infiniteRule = ruleSet->findDoubleRule(uprv_getInfinity());
625 infiniteRule->doFormat(numberToFormat, toInsertInto, _pos + this->pos, recursionCount, status);
626 return;
627 }
b75a7d8f
A
628
629 // if the result is an integer, from here on out we work in integer
630 // space (saving time and memory and preserving accuracy)
2ca993e8
A
631 if (numberToFormat == uprv_floor(numberToFormat) && ruleSet != NULL) {
632 ruleSet->format(util64_fromDouble(numberToFormat), toInsertInto, _pos + this->pos, recursionCount, status);
b75a7d8f
A
633
634 // if the result isn't an integer, then call either our rule set's
635 // format() method or our DecimalFormat's format() method to
636 // format the result
637 } else {
638 if (ruleSet != NULL) {
2ca993e8 639 ruleSet->format(numberToFormat, toInsertInto, _pos + this->pos, recursionCount, status);
73c04bcf 640 } else if (numberFormat != NULL) {
b75a7d8f
A
641 UnicodeString temp;
642 numberFormat->format(numberToFormat, temp);
643 toInsertInto.insert(_pos + this->pos, temp);
644 }
645 }
646}
647
648
649 //-----------------------------------------------------------------------
650 // parsing
651 //-----------------------------------------------------------------------
652
653#ifdef RBNF_DEBUG
654#include <stdio.h>
655#endif
656
657/**
658 * Parses a string using the rule set or DecimalFormat belonging
659 * to this substitution. If there's a match, a mathematical
660 * operation (the inverse of the one used in formatting) is
661 * performed on the result of the parse and the value passed in
662 * and returned as the result. The parse position is updated to
663 * point to the first unmatched character in the string.
664 * @param text The string to parse
665 * @param parsePosition On entry, ignored, but assumed to be 0.
666 * On exit, this is updated to point to the first unmatched
667 * character (or 0 if the substitution didn't match)
668 * @param baseValue A partial parse result that should be
669 * combined with the result of this parse
670 * @param upperBound When searching the rule set for a rule
671 * matching the string passed in, only rules with base values
672 * lower than this are considered
673 * @param lenientParse If true and matching against rules fails,
674 * the substitution will also try matching the text against
675 * numerals using a default-costructed NumberFormat. If false,
676 * no extra work is done. (This value is false whenever the
677 * formatter isn't in lenient-parse mode, but is also false
678 * under some conditions even when the formatter _is_ in
679 * lenient-parse mode.)
680 * @return If there's a match, this is the result of composing
681 * baseValue with whatever was returned from matching the
682 * characters. This will be either a Long or a Double. If there's
683 * no match this is new Long(0) (not null), and parsePosition
684 * is left unchanged.
685 */
686UBool
687NFSubstitution::doParse(const UnicodeString& text,
688 ParsePosition& parsePosition,
689 double baseValue,
690 double upperBound,
691 UBool lenientParse,
692 Formattable& result) const
693{
694#ifdef RBNF_DEBUG
695 fprintf(stderr, "<nfsubs> %x bv: %g ub: %g\n", this, baseValue, upperBound);
696#endif
697 // figure out the highest base value a rule can have and match
698 // the text being parsed (this varies according to the type of
699 // substitutions: multiplier, modulus, and numerator substitutions
700 // restrict the search to rules with base values lower than their
701 // own; same-value substitutions leave the upper bound wherever
702 // it was, and the others allow any rule to match
703 upperBound = calcUpperBound(upperBound);
704
705 // use our rule set to parse the text. If that fails and
706 // lenient parsing is enabled (this is always false if the
707 // formatter's lenient-parsing mode is off, but it may also
708 // be false even when the formatter's lenient-parse mode is
709 // on), then also try parsing the text using a default-
710 // constructed NumberFormat
711 if (ruleSet != NULL) {
712 ruleSet->parse(text, parsePosition, upperBound, result);
713 if (lenientParse && !ruleSet->isFractionRuleSet() && parsePosition.getIndex() == 0) {
714 UErrorCode status = U_ZERO_ERROR;
715 NumberFormat* fmt = NumberFormat::createInstance(status);
716 if (U_SUCCESS(status)) {
717 fmt->parse(text, result, parsePosition);
718 }
719 delete fmt;
720 }
721
722 // ...or use our DecimalFormat to parse the text
73c04bcf 723 } else if (numberFormat != NULL) {
b75a7d8f
A
724 numberFormat->parse(text, result, parsePosition);
725 }
726
727 // if the parse was successful, we've already advanced the caller's
728 // parse position (this is the one function that doesn't have one
729 // of its own). Derive a parse result and return it as a Long,
730 // if possible, or a Double
731 if (parsePosition.getIndex() != 0) {
374ca955
A
732 UErrorCode status = U_ZERO_ERROR;
733 double tempResult = result.getDouble(status);
b75a7d8f
A
734
735 // composeRuleValue() produces a full parse result from
736 // the partial parse result passed to this function from
737 // the caller (this is either the owning rule's base value
738 // or the partial result obtained from composing the
739 // owning rule's base value with its other substitution's
740 // parse result) and the partial parse result obtained by
741 // matching the substitution (which will be the same value
742 // the caller would get by parsing just this part of the
743 // text with RuleBasedNumberFormat.parse() ). How the two
744 // values are used to derive the full parse result depends
745 // on the types of substitutions: For a regular rule, the
746 // ultimate result is its multiplier substitution's result
747 // times the rule's divisor (or the rule's base value) plus
748 // the modulus substitution's result (which will actually
749 // supersede part of the rule's base value). For a negative-
750 // number rule, the result is the negative of its substitution's
751 // result. For a fraction rule, it's the sum of its two
752 // substitution results. For a rule in a fraction rule set,
753 // it's the numerator substitution's result divided by
754 // the rule's base value. Results from same-value substitutions
755 // propagate back upard, and null substitutions don't affect
756 // the result.
757 tempResult = composeRuleValue(tempResult, baseValue);
758 result.setDouble(tempResult);
759 return TRUE;
760 // if the parse was UNsuccessful, return 0
761 } else {
762 result.setLong(0);
763 return FALSE;
764 }
765}
766
b75a7d8f
A
767 /**
768 * Returns true if this is a modulus substitution. (We didn't do this
769 * with instanceof partially because it causes source files to
770 * proliferate and partially because we have to port this to C++.)
771 * @return true if this object is an instance of ModulusSubstitution
772 */
773UBool
774NFSubstitution::isModulusSubstitution() const {
775 return FALSE;
776}
777
729e4ab9 778 /**
2ca993e8 779 * Apple addition
729e4ab9
A
780 * @return true if this is a decimal format-only substitution
781 */
782UBool
783NFSubstitution::isDecimalFormatSubstitutionOnly() const {
784 return (ruleSet == NULL && getNumberFormat() != NULL);
785}
786
787 /**
2ca993e8 788 * Apple addition, not currently used
729e4ab9
A
789 * @return true if this substitution uses another ruleSet
790 */
2ca993e8
A
791//UBool
792//NFSubstitution::isRuleSetSubstitutionOnly() const {
793// return (getNumberFormat() == NULL && ruleSet != NULL);
794//}
729e4ab9 795
b75a7d8f
A
796//===================================================================
797// SameValueSubstitution
798//===================================================================
799
800/**
801 * A substitution that passes the value passed to it through unchanged.
802 * Represented by == in rule descriptions.
803 */
804SameValueSubstitution::SameValueSubstitution(int32_t _pos,
805 const NFRuleSet* _ruleSet,
b75a7d8f
A
806 const UnicodeString& description,
807 UErrorCode& status)
2ca993e8 808: NFSubstitution(_pos, _ruleSet, description, status)
b75a7d8f 809{
4388f060 810 if (0 == description.compare(gEqualsEquals, 2)) {
b75a7d8f
A
811 // throw new IllegalArgumentException("== is not a legal token");
812 status = U_PARSE_ERROR;
813 }
814}
815
46f4442e 816UOBJECT_DEFINE_RTTI_IMPLEMENTATION(SameValueSubstitution)
b75a7d8f
A
817
818//===================================================================
819// MultiplierSubstitution
820//===================================================================
821
46f4442e 822UOBJECT_DEFINE_RTTI_IMPLEMENTATION(MultiplierSubstitution)
b75a7d8f
A
823
824UBool MultiplierSubstitution::operator==(const NFSubstitution& rhs) const
825{
826 return NFSubstitution::operator==(rhs) &&
827 divisor == ((const MultiplierSubstitution*)&rhs)->divisor;
828}
829
830
831//===================================================================
832// ModulusSubstitution
833//===================================================================
834
835/**
836 * A substitution that divides the number being formatted by the its rule's
837 * divisor and formats the remainder. Represented by "&gt;&gt;" in a
838 * regular rule.
839 */
840ModulusSubstitution::ModulusSubstitution(int32_t _pos,
f3c0d7a5 841 const NFRule* rule,
b75a7d8f
A
842 const NFRule* predecessor,
843 const NFRuleSet* _ruleSet,
b75a7d8f
A
844 const UnicodeString& description,
845 UErrorCode& status)
2ca993e8 846 : NFSubstitution(_pos, _ruleSet, description, status)
f3c0d7a5 847 , divisor(rule->getDivisor())
b75a7d8f
A
848 , ruleToUse(NULL)
849{
b75a7d8f
A
850 // the owning rule's divisor controls the behavior of this
851 // substitution: rather than keeping a backpointer to the rule,
852 // we keep a copy of the divisor
853
f3c0d7a5 854 if (divisor == 0) {
374ca955
A
855 status = U_PARSE_ERROR;
856 }
857
4388f060 858 if (0 == description.compare(gGreaterGreaterGreaterThan, 3)) {
b75a7d8f
A
859 // the >>> token doesn't alter how this substituion calculates the
860 // values it uses for formatting and parsing, but it changes
861 // what's done with that value after it's obtained: >>> short-
862 // circuits the rule-search process and goes straight to the
863 // specified rule to format the substitution value
864 ruleToUse = predecessor;
865 }
866}
867
46f4442e 868UOBJECT_DEFINE_RTTI_IMPLEMENTATION(ModulusSubstitution)
b75a7d8f
A
869
870UBool ModulusSubstitution::operator==(const NFSubstitution& rhs) const
871{
872 return NFSubstitution::operator==(rhs) &&
873 divisor == ((const ModulusSubstitution*)&rhs)->divisor &&
874 ruleToUse == ((const ModulusSubstitution*)&rhs)->ruleToUse;
875}
876
877//-----------------------------------------------------------------------
878// formatting
879//-----------------------------------------------------------------------
880
881
882/**
883 * If this is a &gt;&gt;&gt; substitution, use ruleToUse to fill in
884 * the substitution. Otherwise, just use the superclass function.
885 * @param number The number being formatted
886 * @toInsertInto The string to insert the result of this substitution
887 * into
888 * @param pos The position of the rule text in toInsertInto
889 */
890void
2ca993e8 891ModulusSubstitution::doSubstitution(int64_t number, UnicodeString& toInsertInto, int32_t _pos, int32_t recursionCount, UErrorCode& status) const
b75a7d8f
A
892{
893 // if this isn't a >>> substitution, just use the inherited version
894 // of this function (which uses either a rule set or a DecimalFormat
895 // to format its substitution value)
896 if (ruleToUse == NULL) {
2ca993e8 897 NFSubstitution::doSubstitution(number, toInsertInto, _pos, recursionCount, status);
b75a7d8f
A
898
899 // a >>> substitution goes straight to a particular rule to
900 // format the substitution value
901 } else {
902 int64_t numberToFormat = transformNumber(number);
2ca993e8 903 ruleToUse->doFormat(numberToFormat, toInsertInto, _pos + getPos(), recursionCount, status);
b75a7d8f
A
904 }
905}
906
907/**
908* If this is a &gt;&gt;&gt; substitution, use ruleToUse to fill in
909* the substitution. Otherwise, just use the superclass function.
910* @param number The number being formatted
911* @toInsertInto The string to insert the result of this substitution
912* into
913* @param pos The position of the rule text in toInsertInto
914*/
915void
2ca993e8 916ModulusSubstitution::doSubstitution(double number, UnicodeString& toInsertInto, int32_t _pos, int32_t recursionCount, UErrorCode& status) const
b75a7d8f
A
917{
918 // if this isn't a >>> substitution, just use the inherited version
919 // of this function (which uses either a rule set or a DecimalFormat
920 // to format its substitution value)
921 if (ruleToUse == NULL) {
2ca993e8 922 NFSubstitution::doSubstitution(number, toInsertInto, _pos, recursionCount, status);
b75a7d8f
A
923
924 // a >>> substitution goes straight to a particular rule to
925 // format the substitution value
926 } else {
927 double numberToFormat = transformNumber(number);
928
2ca993e8 929 ruleToUse->doFormat(numberToFormat, toInsertInto, _pos + getPos(), recursionCount, status);
b75a7d8f
A
930 }
931}
932
933//-----------------------------------------------------------------------
934// parsing
935//-----------------------------------------------------------------------
936
937/**
938 * If this is a &gt;&gt;&gt; substitution, match only against ruleToUse.
939 * Otherwise, use the superclass function.
940 * @param text The string to parse
941 * @param parsePosition Ignored on entry, updated on exit to point to
942 * the first unmatched character.
943 * @param baseValue The partial parse result prior to calling this
944 * routine.
945 */
946UBool
947ModulusSubstitution::doParse(const UnicodeString& text,
948 ParsePosition& parsePosition,
949 double baseValue,
950 double upperBound,
951 UBool lenientParse,
952 Formattable& result) const
953{
954 // if this isn't a >>> substitution, we can just use the
955 // inherited parse() routine to do the parsing
956 if (ruleToUse == NULL) {
957 return NFSubstitution::doParse(text, parsePosition, baseValue, upperBound, lenientParse, result);
958
959 // but if it IS a >>> substitution, we have to do it here: we
960 // use the specific rule's doParse() method, and then we have to
961 // do some of the other work of NFRuleSet.parse()
962 } else {
963 ruleToUse->doParse(text, parsePosition, FALSE, upperBound, result);
964
965 if (parsePosition.getIndex() != 0) {
73c04bcf
A
966 UErrorCode status = U_ZERO_ERROR;
967 double tempResult = result.getDouble(status);
b75a7d8f
A
968 tempResult = composeRuleValue(tempResult, baseValue);
969 result.setDouble(tempResult);
970 }
971
972 return TRUE;
973 }
974}
4388f060
A
975/**
976 * Returns a textual description of the substitution
977 * @return A textual description of the substitution. This might
978 * not be identical to the description it was created from, but
979 * it'll produce the same result.
980 */
981void
982ModulusSubstitution::toString(UnicodeString& text) const
983{
984 // use tokenChar() to get the character at the beginning and
985 // end of the substitutin token. In between them will go
986 // either the name of the rule set it uses, or the pattern of
987 // the DecimalFormat it uses
b75a7d8f 988
4388f060
A
989 if ( ruleToUse != NULL ) { // Must have been a >>> substitution.
990 text.remove();
991 text.append(tokenChar());
992 text.append(tokenChar());
993 text.append(tokenChar());
994 } else { // Otherwise just use the super-class function.
995 NFSubstitution::toString(text);
996 }
997}
b75a7d8f
A
998//===================================================================
999// IntegralPartSubstitution
1000//===================================================================
1001
46f4442e 1002UOBJECT_DEFINE_RTTI_IMPLEMENTATION(IntegralPartSubstitution)
b75a7d8f
A
1003
1004
1005//===================================================================
1006// FractionalPartSubstitution
1007//===================================================================
1008
1009
1010 /**
1011 * Constructs a FractionalPartSubstitution. This object keeps a flag
1012 * telling whether it should format by digits or not. In addition,
1013 * it marks the rule set it calls (if any) as a fraction rule set.
1014 */
1015FractionalPartSubstitution::FractionalPartSubstitution(int32_t _pos,
1016 const NFRuleSet* _ruleSet,
b75a7d8f
A
1017 const UnicodeString& description,
1018 UErrorCode& status)
2ca993e8 1019 : NFSubstitution(_pos, _ruleSet, description, status)
b75a7d8f
A
1020 , byDigits(FALSE)
1021 , useSpaces(TRUE)
1022
1023{
1024 // akk, ruleSet can change in superclass constructor
4388f060
A
1025 if (0 == description.compare(gGreaterGreaterThan, 2) ||
1026 0 == description.compare(gGreaterGreaterGreaterThan, 3) ||
b75a7d8f
A
1027 _ruleSet == getRuleSet()) {
1028 byDigits = TRUE;
4388f060 1029 if (0 == description.compare(gGreaterGreaterGreaterThan, 3)) {
b75a7d8f
A
1030 useSpaces = FALSE;
1031 }
1032 } else {
1033 // cast away const
1034 ((NFRuleSet*)getRuleSet())->makeIntoFractionRuleSet();
1035 }
1036}
1037
1038//-----------------------------------------------------------------------
1039// formatting
1040//-----------------------------------------------------------------------
1041
1042/**
1043 * If in "by digits" mode, fills in the substitution one decimal digit
1044 * at a time using the rule set containing this substitution.
1045 * Otherwise, uses the superclass function.
1046 * @param number The number being formatted
1047 * @param toInsertInto The string to insert the result of formatting
1048 * the substitution into
1049 * @param pos The position of the owning rule's rule text in
1050 * toInsertInto
1051 */
1052void
b331163b 1053FractionalPartSubstitution::doSubstitution(double number, UnicodeString& toInsertInto,
2ca993e8 1054 int32_t _pos, int32_t recursionCount, UErrorCode& status) const
b75a7d8f 1055{
374ca955
A
1056 // if we're not in "byDigits" mode, just use the inherited
1057 // doSubstitution() routine
1058 if (!byDigits) {
2ca993e8 1059 NFSubstitution::doSubstitution(number, toInsertInto, _pos, recursionCount, status);
374ca955
A
1060
1061 // if we're in "byDigits" mode, transform the value into an integer
1062 // by moving the decimal point eight places to the right and
1063 // pulling digits off the right one at a time, formatting each digit
1064 // as an integer using this substitution's owning rule set
1065 // (this is slower, but more accurate, than doing it from the
1066 // other end)
1067 } else {
1068 // int32_t numberToFormat = (int32_t)uprv_round(transformNumber(number) * uprv_pow(10, kMaxDecimalDigits));
1069 // // this flag keeps us from formatting trailing zeros. It starts
1070 // // out false because we're pulling from the right, and switches
1071 // // to true the first time we encounter a non-zero digit
1072 // UBool doZeros = FALSE;
1073 // for (int32_t i = 0; i < kMaxDecimalDigits; i++) {
1074 // int64_t digit = numberToFormat % 10;
1075 // if (digit != 0 || doZeros) {
1076 // if (doZeros && useSpaces) {
1077 // toInsertInto.insert(_pos + getPos(), gSpace);
1078 // }
1079 // doZeros = TRUE;
1080 // getRuleSet()->format(digit, toInsertInto, _pos + getPos());
1081 // }
1082 // numberToFormat /= 10;
1083 // }
1084
1085 DigitList dl;
729e4ab9
A
1086 dl.set(number);
1087 dl.roundFixedPoint(20); // round to 20 fraction digits.
1088 dl.reduce(); // Removes any trailing zeros.
374ca955
A
1089
1090 UBool pad = FALSE;
729e4ab9
A
1091 for (int32_t didx = dl.getCount()-1; didx>=dl.getDecimalAt(); didx--) {
1092 // Loop iterates over fraction digits, starting with the LSD.
1093 // include both real digits from the number, and zeros
1094 // to the left of the MSD but to the right of the decimal point.
374ca955
A
1095 if (pad && useSpaces) {
1096 toInsertInto.insert(_pos + getPos(), gSpace);
1097 } else {
1098 pad = TRUE;
1099 }
729e4ab9 1100 int64_t digit = didx>=0 ? dl.getDigit(didx) - '0' : 0;
2ca993e8 1101 getRuleSet()->format(digit, toInsertInto, _pos + getPos(), recursionCount, status);
374ca955 1102 }
b75a7d8f 1103
374ca955
A
1104 if (!pad) {
1105 // hack around lack of precision in digitlist. if we would end up with
1106 // "foo point" make sure we add a " zero" to the end.
2ca993e8 1107 getRuleSet()->format((int64_t)0, toInsertInto, _pos + getPos(), recursionCount, status);
b75a7d8f 1108 }
374ca955 1109 }
b75a7d8f
A
1110}
1111
1112//-----------------------------------------------------------------------
1113// parsing
1114//-----------------------------------------------------------------------
1115
1116/**
1117 * If in "by digits" mode, parses the string as if it were a string
1118 * of individual digits; otherwise, uses the superclass function.
1119 * @param text The string to parse
1120 * @param parsePosition Ignored on entry, but updated on exit to point
1121 * to the first unmatched character
1122 * @param baseValue The partial parse result prior to entering this
1123 * function
1124 * @param upperBound Only consider rules with base values lower than
1125 * this when filling in the substitution
1126 * @param lenientParse If true, try matching the text as numerals if
1127 * matching as words doesn't work
1128 * @return If the match was successful, the current partial parse
1129 * result; otherwise new Long(0). The result is either a Long or
1130 * a Double.
1131 */
1132
1133UBool
1134FractionalPartSubstitution::doParse(const UnicodeString& text,
1135 ParsePosition& parsePosition,
1136 double baseValue,
1137 double /*upperBound*/,
1138 UBool lenientParse,
1139 Formattable& resVal) const
1140{
1141 // if we're not in byDigits mode, we can just use the inherited
1142 // doParse()
1143 if (!byDigits) {
1144 return NFSubstitution::doParse(text, parsePosition, baseValue, 0, lenientParse, resVal);
1145
1146 // if we ARE in byDigits mode, parse the text one digit at a time
1147 // using this substitution's owning rule set (we do this by setting
1148 // upperBound to 10 when calling doParse() ) until we reach
1149 // nonmatching text
1150 } else {
1151 UnicodeString workText(text);
1152 ParsePosition workPos(1);
1153 double result = 0;
1154 int32_t digit;
374ca955 1155// double p10 = 0.1;
b75a7d8f 1156
374ca955 1157 DigitList dl;
b75a7d8f
A
1158 NumberFormat* fmt = NULL;
1159 while (workText.length() > 0 && workPos.getIndex() != 0) {
1160 workPos.setIndex(0);
1161 Formattable temp;
1162 getRuleSet()->parse(workText, workPos, 10, temp);
374ca955
A
1163 UErrorCode status = U_ZERO_ERROR;
1164 digit = temp.getLong(status);
1165// digit = temp.getType() == Formattable::kLong ?
1166// temp.getLong() :
1167// (int32_t)temp.getDouble();
b75a7d8f
A
1168
1169 if (lenientParse && workPos.getIndex() == 0) {
1170 if (!fmt) {
374ca955 1171 status = U_ZERO_ERROR;
b75a7d8f
A
1172 fmt = NumberFormat::createInstance(status);
1173 if (U_FAILURE(status)) {
1174 delete fmt;
1175 fmt = NULL;
1176 }
1177 }
1178 if (fmt) {
1179 fmt->parse(workText, temp, workPos);
73c04bcf 1180 digit = temp.getLong(status);
b75a7d8f
A
1181 }
1182 }
1183
1184 if (workPos.getIndex() != 0) {
374ca955
A
1185 dl.append((char)('0' + digit));
1186// result += digit * p10;
1187// p10 /= 10;
b75a7d8f
A
1188 parsePosition.setIndex(parsePosition.getIndex() + workPos.getIndex());
1189 workText.removeBetween(0, workPos.getIndex());
1190 while (workText.length() > 0 && workText.charAt(0) == gSpace) {
1191 workText.removeBetween(0, 1);
1192 parsePosition.setIndex(parsePosition.getIndex() + 1);
1193 }
1194 }
1195 }
1196 delete fmt;
1197
729e4ab9 1198 result = dl.getCount() == 0 ? 0 : dl.getDouble();
b75a7d8f
A
1199 result = composeRuleValue(result, baseValue);
1200 resVal.setDouble(result);
1201 return TRUE;
1202 }
1203}
1204
1205UBool
1206FractionalPartSubstitution::operator==(const NFSubstitution& rhs) const
1207{
1208 return NFSubstitution::operator==(rhs) &&
1209 ((const FractionalPartSubstitution*)&rhs)->byDigits == byDigits;
1210}
1211
46f4442e 1212UOBJECT_DEFINE_RTTI_IMPLEMENTATION(FractionalPartSubstitution)
b75a7d8f
A
1213
1214
1215//===================================================================
1216// AbsoluteValueSubstitution
1217//===================================================================
1218
46f4442e 1219UOBJECT_DEFINE_RTTI_IMPLEMENTATION(AbsoluteValueSubstitution)
b75a7d8f
A
1220
1221//===================================================================
1222// NumeratorSubstitution
1223//===================================================================
1224
73c04bcf 1225void
2ca993e8 1226NumeratorSubstitution::doSubstitution(double number, UnicodeString& toInsertInto, int32_t apos, int32_t recursionCount, UErrorCode& status) const {
73c04bcf
A
1227 // perform a transformation on the number being formatted that
1228 // is dependent on the type of substitution this is
1229
1230 double numberToFormat = transformNumber(number);
1231 int64_t longNF = util64_fromDouble(numberToFormat);
1232
1233 const NFRuleSet* aruleSet = getRuleSet();
1234 if (withZeros && aruleSet != NULL) {
1235 // if there are leading zeros in the decimal expansion then emit them
1236 int64_t nf =longNF;
1237 int32_t len = toInsertInto.length();
1238 while ((nf *= 10) < denominator) {
1239 toInsertInto.insert(apos + getPos(), gSpace);
2ca993e8 1240 aruleSet->format((int64_t)0, toInsertInto, apos + getPos(), recursionCount, status);
73c04bcf
A
1241 }
1242 apos += toInsertInto.length() - len;
1243 }
1244
1245 // if the result is an integer, from here on out we work in integer
1246 // space (saving time and memory and preserving accuracy)
1247 if (numberToFormat == longNF && aruleSet != NULL) {
2ca993e8 1248 aruleSet->format(longNF, toInsertInto, apos + getPos(), recursionCount, status);
73c04bcf
A
1249
1250 // if the result isn't an integer, then call either our rule set's
1251 // format() method or our DecimalFormat's format() method to
1252 // format the result
1253 } else {
1254 if (aruleSet != NULL) {
2ca993e8 1255 aruleSet->format(numberToFormat, toInsertInto, apos + getPos(), recursionCount, status);
73c04bcf 1256 } else {
73c04bcf
A
1257 UnicodeString temp;
1258 getNumberFormat()->format(numberToFormat, temp, status);
1259 toInsertInto.insert(apos + getPos(), temp);
1260 }
1261 }
1262}
1263
1264UBool
1265NumeratorSubstitution::doParse(const UnicodeString& text,
1266 ParsePosition& parsePosition,
1267 double baseValue,
1268 double upperBound,
1269 UBool /*lenientParse*/,
1270 Formattable& result) const
1271{
1272 // we don't have to do anything special to do the parsing here,
1273 // but we have to turn lenient parsing off-- if we leave it on,
1274 // it SERIOUSLY messes up the algorithm
1275
1276 // if withZeros is true, we need to count the zeros
1277 // and use that to adjust the parse result
1278 UErrorCode status = U_ZERO_ERROR;
1279 int32_t zeroCount = 0;
1280 UnicodeString workText(text);
1281
1282 if (withZeros) {
1283 ParsePosition workPos(1);
1284 Formattable temp;
1285
1286 while (workText.length() > 0 && workPos.getIndex() != 0) {
1287 workPos.setIndex(0);
1288 getRuleSet()->parse(workText, workPos, 1, temp); // parse zero or nothing at all
1289 if (workPos.getIndex() == 0) {
1290 // we failed, either there were no more zeros, or the number was formatted with digits
1291 // either way, we're done
1292 break;
1293 }
1294
1295 ++zeroCount;
1296 parsePosition.setIndex(parsePosition.getIndex() + workPos.getIndex());
1297 workText.remove(0, workPos.getIndex());
1298 while (workText.length() > 0 && workText.charAt(0) == gSpace) {
1299 workText.remove(0, 1);
1300 parsePosition.setIndex(parsePosition.getIndex() + 1);
1301 }
1302 }
1303
1304 workText = text;
1305 workText.remove(0, (int32_t)parsePosition.getIndex());
1306 parsePosition.setIndex(0);
1307 }
1308
1309 // we've parsed off the zeros, now let's parse the rest from our current position
1310 NFSubstitution::doParse(workText, parsePosition, withZeros ? 1 : baseValue, upperBound, FALSE, result);
1311
1312 if (withZeros) {
1313 // any base value will do in this case. is there a way to
1314 // force this to not bother trying all the base values?
1315
1316 // compute the 'effective' base and prescale the value down
1317 int64_t n = result.getLong(status); // force conversion!
1318 int64_t d = 1;
1319 int32_t pow = 0;
1320 while (d <= n) {
1321 d *= 10;
1322 ++pow;
1323 }
1324 // now add the zeros
1325 while (zeroCount > 0) {
1326 d *= 10;
1327 --zeroCount;
1328 }
1329 // d is now our true denominator
1330 result.setDouble((double)n/(double)d);
1331 }
1332
1333 return TRUE;
1334}
1335
b75a7d8f
A
1336UBool
1337NumeratorSubstitution::operator==(const NFSubstitution& rhs) const
1338{
1339 return NFSubstitution::operator==(rhs) &&
1340 denominator == ((const NumeratorSubstitution*)&rhs)->denominator;
1341}
1342
46f4442e 1343UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NumeratorSubstitution)
b75a7d8f 1344
73c04bcf
A
1345const UChar NumeratorSubstitution::LTLT[] = { 0x003c, 0x003c };
1346
46f4442e 1347U_NAMESPACE_END
b75a7d8f
A
1348
1349/* U_HAVE_RBNF */
1350#endif
1351