Update contrib.
2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
19 #include "NumberConversion.h"
20 #include "NumberConversionImp.h"
24 ENumConvPanicInvalidDigitType = 1,
28 LOCAL_D void Panic(TNumConvPanic aPanicCode)
30 // Panic the thread with NumberConversion as the category
33 _LIT(KPanicNumConv, "NumConv");
34 User::Panic(KPanicNumConv, aPanicCode);
38 // TStandardDigitMatch
41 Length required to format KMinTInt in the longest number format
44 const TInt KMaxLengthOfFormattedNumber = 11;
46 _LIT(KDefaultDigit,"0");
47 _LIT(KFormatIdentifier,"%");
49 TInt StandardDigitMatch::Match(const TDesC& aText, TInt& aLength,
50 TDigitType& aDigitType, NumberConversion::TDigitMatchType aDigitMatchType)
54 aDigitType = EDigitTypeUnknown;
55 TDigitType currentDigitType = EDigitTypeUnknown;
57 TInt textLength = aText.Length();
58 while (aLength < textLength)
60 TChar currentChar = aText[aLength];
61 currentDigitType = DigitType(currentChar);
62 if (currentDigitType == EDigitTypeUnknown)
67 TUint charValue = currentChar;
68 digit = charValue - currentDigitType;
69 if (aDigitType == EDigitTypeUnknown)
71 aDigitType = currentDigitType;
75 if (aDigitType != currentDigitType)
77 if (aDigitMatchType == NumberConversion::EMatchMultipleTypes)
79 aDigitType = EDigitTypeAllTypes;
88 total = (total * 10) + digit;
94 TInt StandardDigitMatch::LeadingZeros(const TDesC& aText)
96 //Function to find the number of leading zeros
97 TInt textLength = aText.Length();
98 TInt numOfLeadingZeros = 0;
99 TInt currentLength = 0;
100 TDigitType currentDigitType = EDigitTypeUnknown;
101 TBool leadingZeros = ETrue;
104 return numOfLeadingZeros; // No leading number, if only one number.
106 while(currentLength < textLength)
108 TChar currentChar = aText[currentLength];
109 currentDigitType = DigitType(currentChar);
110 if (currentDigitType == EDigitTypeUnknown)
112 return numOfLeadingZeros;
115 TUint charValue = currentChar;
116 digit = charValue - currentDigitType;
118 if (digit!=0 && leadingZeros)
124 return numOfLeadingZeros;
128 TDigitType StandardDigitMatch::DigitType(TChar aChar)
130 if (aChar >= EDigitTypeWestern && aChar < EDigitTypeWestern+10)
132 return EDigitTypeWestern;
134 if (aChar >= EDigitTypeArabicIndic && aChar < EDigitTypeArabicIndic+10)
136 return EDigitTypeArabicIndic;
138 if (aChar >= EDigitTypeEasternArabicIndic
139 && aChar < EDigitTypeEasternArabicIndic+10)
141 return EDigitTypeEasternArabicIndic;
143 if (aChar >= EDigitTypeDevanagari && aChar < EDigitTypeDevanagari+10)
145 return EDigitTypeDevanagari;
147 if (aChar >= EDigitTypeBengali && aChar < EDigitTypeBengali+10)
149 return EDigitTypeBengali;
151 if (aChar >= EDigitTypeGurmukhi && aChar < EDigitTypeGurmukhi+10)
153 return EDigitTypeGurmukhi;
155 if (aChar >= EDigitTypeGujarati && aChar < EDigitTypeGujarati+10)
157 return EDigitTypeGujarati;
159 if (aChar >= EDigitTypeOriya && aChar < EDigitTypeOriya+10)
161 return EDigitTypeOriya;
163 if (aChar >= EDigitTypeTamil && aChar < EDigitTypeTamil+10)
165 return EDigitTypeTamil;
167 if (aChar >= EDigitTypeTelugu && aChar < EDigitTypeTelugu+10)
169 return EDigitTypeTelugu;
171 if (aChar >= EDigitTypeKannada && aChar < EDigitTypeKannada+10)
173 return EDigitTypeKannada;
175 if (aChar >= EDigitTypeMalayalam && aChar < EDigitTypeMalayalam+10)
177 return EDigitTypeMalayalam;
179 if (aChar >= EDigitTypeThai && aChar < EDigitTypeThai+10)
181 return EDigitTypeThai;
183 if (aChar >= EDigitTypeLao && aChar < EDigitTypeLao+10)
185 return EDigitTypeLao;
187 if (aChar >= EDigitTypeTibetan && aChar < EDigitTypeTibetan+10)
189 return EDigitTypeTibetan;
191 if (aChar >= EDigitTypeTibetan && aChar < EDigitTypeTibetan+20)
193 return EDigitTypeTibetan;
195 if (aChar >= EDigitTypeMayanmar && aChar < EDigitTypeMayanmar+10)
197 return EDigitTypeMayanmar;
199 if (aChar >= EDigitTypeKhmer && aChar < EDigitTypeKhmer+10)
201 return EDigitTypeKhmer;
203 return EDigitTypeUnknown;
206 void StandardDigitMatch::AppendFormat(TDes& aText, TInt aNumber,
207 TDigitType aDigitType)
209 TInt base = aDigitType;
211 if (base != EDigitTypeUnknown)
213 TInt length = aText.Length();
214 TInt number = aNumber;
215 TBuf<1> digitText(KDefaultDigit);
218 digitText[0] = (TUint16)((number % 10) + base);
219 aText.Insert(length, digitText);
221 } while (number > 0);
225 TInt StandardDigitMatch::LengthOfFormattedNumber(TInt aNumber)
232 } while (aNumber > 0);
241 EXPORT_C TInt NumberConversion::ConvertFirstNumber(const TDesC& aText,
242 TInt& aLength, TDigitType& aDigitType, TDigitMatchType aDigitMatchType)
244 Converts the descriptor aText into an integer and returns it. Ignores any minus
245 signs: only non-negative numbers are returned.
247 @param aText Input text containing the integer to be converted.
248 @param aLength On exit aLength is set to the number of characters converted
250 @param aDigitType Returns the digit type of the number converted. If no
251 characters are matched, aDigitType will be set to EDigitTypeUnknown.
252 @param aDigitMatchType If aDigitMatchType is set to EMatchMultipleTypes,
253 different number types in the descriptor are matched and returned as a single
254 number. In this case, aDigitType will be set to EDigitTypeAllTypes.
255 @return The (non-negative) number found.
258 return StandardDigitMatch::Match(aText, aLength, aDigitType, aDigitMatchType);
261 EXPORT_C TInt NumberConversion::PositionAndTypeOfNextNumber(const TDesC& aText,
262 TDigitType& aDigitType, TInt aStartFrom)
264 Finds the position and type of the next number in the descriptor. If the number
265 has a preceeding minus sign, it will be ignored and the position of the first
266 digit will be returned.
268 @param aText Text to be searched.
269 @param aStartFrom First position within aText to be searched.
270 @param aDigitType aDigitType is set to the digit type matched. If the
271 descriptor doesn't contain a recognisable digit, aDigitType is set to
273 @return The index of the first character.
276 TInt index = aStartFrom;
277 TInt length = aText.Length();
278 while (index < length)
280 TChar currentChar(aText[index]);
281 aDigitType = StandardDigitMatch::DigitType(currentChar);
282 if (aDigitType != EDigitTypeUnknown)
291 EXPORT_C void NumberConversion::FormatNumber(TDes& aText, TInt aNumber,
292 TDigitType aDigitType)
294 Converts a non-negative integer into localised text.
295 @param aText Output for the converted number. aText must be long enough to
296 accommodate the text or the descriptor will panic. Negative numbers are not
298 @param aNumber The number to be converted.
299 @param aDigitType The type of digit to render the number in.
300 @pre NumberConversion::LengthOfFormattedNumber(aNumber, aDigitType) <= aText.MaxLength() && 0 <= aNumber
304 AppendFormatNumber(aText, aNumber, aDigitType);
307 EXPORT_C void NumberConversion::FormatDigit(TDes& aText, TInt aNumber, TInt aLeadingZero,
308 TDigitType aDigitType)
310 Converts a non-negative integer into localised text.
312 @param aText Output for the converted number. aText must be long enough to
313 accommodate the text or the descriptor will panic. Negative numbers are not
315 @param aNumber The number to be converted.
316 @param aDigitType The type of digit to render the number in.
317 @param aLeadingZero The number of zeros that appear before the number to be
318 converted to localised text.
319 @pre NumberConversion::LengthOfFormattedNumber(aNumber, aDigitType) <= aText.MaxLength() && 0 <= aNumber
323 AppendFormatNumber(aText, aNumber, aDigitType);
328 TChar zero = aDigitType;
330 zeroBuf.Append(zero);
331 for (TInt i=0; i<aLeadingZero; i++)
333 aText.Insert(0,zeroBuf);
338 EXPORT_C void NumberConversion::AppendFormatNumber(TDes& aText, TInt aNumber,
339 TDigitType aDigitType)
341 Converts a non-negative integer into localised text, appending the result to a
344 @param aText Output for the converted number. aText must have enough free space
345 after its current length to accommodate the text or the descriptor will panic.
346 Negative numbers are not supported.
347 @param aNumber The number to be converted.
348 @param aDigitType The type of digit to render the number in.
349 @pre NumberConversion::LengthOfFormattedNumber(aNumber, aDigitType) <= aText.MaxLength() - aText.Length() && 0 <= aNumber
352 StandardDigitMatch::AppendFormat(aText, aNumber, aDigitType);
355 EXPORT_C void NumberConversion::ConvertDigits(TDes& aText,
356 TDigitType aDigitType)
358 Converts all of the digits in the descriptor aText into the format specified in
361 @param aText The text that is to have its digits converted.
362 @param aDigitType The digit type to be converted to.
363 @pre NumberConversion::LengthOfConvertedText(aText, aDigitType) <= aText.MaxLength()
364 @post All digits in the string will either conform to one of the constants
365 defined in enum TDigitType or will match the digit type
366 supplied in aDigitType.
370 for (TInt i=0; i < aText.Length(); i++)
372 if ( (charValue = static_cast<TChar>(aText[i]).GetNumericValue()) >= 0)
374 TBuf<1> convertedNumber;
375 FormatNumber(convertedNumber, charValue, aDigitType);
377 aText.Insert(i, convertedNumber);
382 EXPORT_C TInt NumberConversion::LengthOfFormattedNumber(TInt aNumber,
383 TDigitType /*aDigitType*/)
385 Returns the number of characters required to format aNumber into text.
386 @param aNumber The number to be converted.
387 @param aDigitType The format for the number.
388 @return The length of descriptor required to render the number as text.
391 return StandardDigitMatch::LengthOfFormattedNumber(aNumber);
394 EXPORT_C TInt NumberConversion::LengthOfConvertedText(const TDesC& aText,
395 TDigitType aDigitType)
397 Returns the length of the descriptor required to hold text with its digits
400 @param aText Input text for the conversion.
401 @param aDigitType The type of digit that will be used for the conversion.
402 @return The length of descriptor that would be required to convert the digits
403 in aText into the type specified by aDigitType.
406 TDigitType digitType = EDigitTypeUnknown;
407 TInt position = PositionAndTypeOfNextNumber(aText, digitType);
408 TInt total = aText.Length();
409 while (digitType != EDigitTypeUnknown)
411 // Convert this number into a different format
413 TPtrC matchText = aText.Mid(position);
414 TInt number = ConvertFirstNumber(matchText, length, digitType);
416 total += LengthOfFormattedNumber(number, aDigitType);
417 position = PositionAndTypeOfNextNumber(aText, digitType,
423 EXPORT_C void NumberConversion::Format(TDigitType aDigitType,
424 TRefByValue<TDes> aFmt,...)
426 Formats the descriptor. Format specifiers are converted to values passed in the
427 variable argument list. The following format specifiers are supported:
429 %d - Interprets the argument as a TInt and formats it using the aDigitType
430 format. Negative numbers are not supported.
432 %S - Interprets the argument as a pointer to a TDesC and inserts it into the
435 @param aDigitType The digit type for all %d directives.
438 VA_LIST argument_list;
439 VA_START(argument_list, aFmt);
442 TInt match = KErrNotFound;
443 while ((match = format.Find(KFormatIdentifier)) != KErrNotFound)
445 TChar formatIdentifier = format[match+1];
446 switch (formatIdentifier)
450 format.Delete(match,2);
451 TInt number = VA_ARG(argument_list, int);
452 TBuf<KMaxLengthOfFormattedNumber> convertedNumber;
453 FormatNumber(convertedNumber, number, aDigitType);
454 format.Insert(match, convertedNumber);
459 format.Delete(match,2);
460 TDesC* des = VA_ARG(argument_list, TDesC*);
461 format.Insert(match, *des);
465 // Remove format identifier
466 format.Delete(match,1);
470 VA_END(argument_list);
473 EXPORT_C TChar NumberConversion::ConvertDigit(TChar& aDigit, TDigitType aDigitType)
475 converts aDigit (which could be arabic, western digit etc) into the form
476 denoted by aDigitType.
478 @param TChar& aDigit - contains the digit to be converted.
479 @param TDigitType aDigitType - aDigit type: western, arabic, thai, ...
480 @return Digit into the form, denoted by aDigitType.
483 __ASSERT_DEBUG(EDigitTypeUnknown != aDigitType && EDigitTypeAllTypes != aDigitType,
484 Panic(ENumConvPanicInvalidDigitType));
487 NumberConversion::ConvertDigits(buf, aDigitType);