Update contrib.
2 * Copyright (c) 2002-2008 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of the License "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.
20 #include "NumberGrouping.h"
21 #include "RegularExpression.h"
22 #include "cleanuputil.h"
26 #include <centralrepository.h>
27 #include <NumberGroupingCRKeys.h>
29 const TText KNumberGroupingWildcard('n');
30 const TText KNumberGroupingOneOrMoreCharactersToken('~');
32 // This constant represents all the valid whitespace that we can handle. Several
33 // APIs give whitespace special significance as a formatting character, especially
34 // those methods used to obtain text for formatting into fixed-width UI elements
35 // where spaces are not to be rendered at the margins.
36 const TText KNumberGroupingSpace(' ');
38 const TInt KMinimumLengthToGroup = 1; // No grouping occurs if fewer than this in unformatted buffer
40 #include <numbergrouping.rsg>
43 GLDEF_C void Panic(TNumberGroupingPanic aPanic)
45 _LIT(KPanicCat,"Number Grouping");
46 User::Panic(KPanicCat, aPanic);
49 // Valid phone number characters apart from IsDigit() characters are listed here
50 const TText KAdditionalPhoneNumberCharacters[] = {'+','#','*','p','w'};
55 NONSHARABLE_CLASS(CPNGNumberGroupingExtension): public CBase
58 CPNGNumberGroupingExtension();
59 ~CPNGNumberGroupingExtension();
61 TInt iMaxExtraCharacters;
62 TInt iNumberGroupingCRValue;
65 CPNGNumberGroupingExtension::CPNGNumberGroupingExtension()
69 CPNGNumberGroupingExtension::~CPNGNumberGroupingExtension()
75 CPNGNumberGrouping::TPNGSeparator::TPNGSeparator()
76 : iPosition(-1), iSeparatorCharacter(KNumberGroupingSpace)
80 CPNGNumberGrouping::TPNGSeparator::TPNGSeparator( TInt aPosition, TText aSeparatorCharacter )
81 : iPosition(aPosition), iSeparatorCharacter(aSeparatorCharacter)
85 CPNGNumberGrouping::TPNGGroupingInfo::TPNGGroupingInfo()
89 // CPNGNumberGrouping - grouping engine class
90 CPNGNumberGrouping::CPNGNumberGrouping(TInt aMaxLength /* = 0 */, TBool aReversed /* = EFalse */) :
91 iForceLanguage(ELangTest),
92 iUnformattedNumberPtr(KNullDesC),
93 iFormattedNumberPtr(KNullDesC),
94 iReverseFormattedNumberPtr(KNullDesC),
95 iSelectionPtr(KNullDesC),
97 iMaxUnformattedLength(aMaxLength),
99 iMatchedPatternIndex(ENoMatchedPattern)
103 EXPORT_C CPNGNumberGrouping* CPNGNumberGrouping::NewL( TInt aMaxLength, TBool aReserved)
105 CPNGNumberGrouping* s = NewLC(aMaxLength, aReserved);
110 EXPORT_C CPNGNumberGrouping* CPNGNumberGrouping::NewLC( TInt aMaxLength, TBool aReserved)
112 CPNGNumberGrouping* s = new(ELeave)CPNGNumberGrouping( aMaxLength, aReserved);
113 CleanupStack::PushL(s);
118 void CPNGNumberGrouping::ConstructL()
120 iExtension = new (ELeave) CPNGNumberGroupingExtension();
121 CRepository* repository = NULL;
122 iExtension->iNumberGroupingCRValue = 0;
123 TRAPD(ret, repository = CRepository::NewL(KCRUidNumberGrouping));
126 ret = repository->Get(KNumberGrouping, iExtension->iNumberGroupingCRValue);
130 // Read from resource first in order to obtain a value for iExtension->iMaxExtraCharacters
131 iLanguage = doReadLanguageFromSharedData();
132 doReadFormatInfoFromResourceFileL(); // This sets iExtension->iMaxExtraCharacters
134 // Allocation of buffers. Note that iMaxUnformattedLength must be retained as member
135 // data because HBufCs may come back with more storage available than asked for.
136 // The uncertainty in the actual length provided by the HBufCs means that although
137 // they are asked to be different by iExtension->iMaxExtraCharacters, the difference between the
138 // actual MaxLengths may be less than iMaxExtraCharcaters.
139 // The approach decided upon is to retain iMaxUnformattedLength as member data and
140 // use IT everywhere throughout this class's implementation, NEVER using
141 // iUnformattedNumber->Des().MaxLength()
142 iUnformattedNumber = HBufC::NewL(iMaxUnformattedLength);
143 iFormattedNumber = HBufC::NewL(iMaxUnformattedLength + iExtension->iMaxExtraCharacters);
145 // Create revesed buffer only if requested
147 iReverseFormattedNumber = HBufC::NewL(iMaxUnformattedLength + iExtension->iMaxExtraCharacters);
150 EXPORT_C CPNGNumberGrouping::~CPNGNumberGrouping()
152 doClearGroupingItemsList();
154 delete iUnformattedNumber;
155 delete iFormattedNumber;
156 delete iReverseFormattedNumber;
161 EXPORT_C TInt CPNGNumberGrouping::Insert(TInt aIndex, TText aChar)
164 if( aIndex >= 0 && aIndex <= iUnformattedNumber->Length())
167 if(iUnformattedNumber->Length() >= iMaxUnformattedLength)
172 TPtr ptrModifyable(iUnformattedNumber->Des());
173 ptrModifyable.Insert(aIndex, bufChar);
175 doClearFormattedNumbers();
180 return KErrIndexOutOfRange;
183 EXPORT_C TInt CPNGNumberGrouping::Delete(TInt aIndex)
185 if(aIndex >= 0 && aIndex < iUnformattedNumber->Length())
187 TPtr ptrModifyable(iUnformattedNumber->Des());
188 ptrModifyable.Delete(aIndex, KSingleCharacter);
190 doClearFormattedNumbers();
195 return KErrIndexOutOfRange;
198 EXPORT_C TInt CPNGNumberGrouping::Append(TText aChar)
200 if(iUnformattedNumber->Length() >= iMaxUnformattedLength)
205 TPtr ptrModifyable(iUnformattedNumber->Des());
206 ptrModifyable.Append(bufChar);
208 doClearFormattedNumbers();
213 EXPORT_C TInt CPNGNumberGrouping::Set(const TDesC& aNumber)
215 if( aNumber.Length() > iMaxUnformattedLength )
218 TPtr ptrModifyable( iUnformattedNumber->Des() );
219 ptrModifyable.Copy( aNumber );
221 doClearFormattedNumbers();
226 EXPORT_C TInt CPNGNumberGrouping::Length() const
228 if(!iFormattedNumber->Length()) // This test is used as the trigger to reformat
231 return iFormattedNumber->Length();
234 EXPORT_C TInt CPNGNumberGrouping::UnFormattedLength() const
236 return iUnformattedNumber->Length();
239 EXPORT_C TInt CPNGNumberGrouping::MaxDisplayLength() const
241 // Despite its name, this method returns the max length of the UNFORMATTED buffer
242 // This must not be implemented to return the actual length available in
243 // iUnformattedBuffer.
244 return iMaxUnformattedLength;
247 EXPORT_C TBool CPNGNumberGrouping::IsSpace(TInt aPos) const
249 // Very tricky semantics for this. Must be a space inserted by the formatting
250 if ( iFormattedNumber->Length() > aPos &&
251 iFormattedNumber->operator[](aPos) == KNumberGroupingSpace &&
252 // Check also that is is less than the length to group + inserted characters
253 aPos < ( LengthToGroup() + (iFormattedNumber->Length() - iUnformattedNumber->Length()) ) )
259 EXPORT_C const TDesC& CPNGNumberGrouping::FormattedNumber(TInt aFrom, TInt aTo) const
261 if(iUnformattedNumber->Length() != 0 &&
266 iFormattedNumberPtr.Set(KNullDesC);
268 TInt length = iFormattedNumber->Length();
271 TInt length = iFormattedNumber->Length();
272 if ( iExtension->iNumberGroupingCRValue )
274 // Advance to the next non-space
275 while( (aFrom < length ) && (*iFormattedNumber)[aFrom] == KNumberGroupingSpace )
280 // Retreat to the last non-space
281 while( (aTo > 0) && (*iFormattedNumber)[aTo] == KNumberGroupingSpace )
287 // Does fetching the descriptor still make sense?
288 if ( (0 <= aFrom) && (aFrom <= aTo) && (aTo < length) )
289 iFormattedNumberPtr.Set( iFormattedNumber->Mid( aFrom, aTo-aFrom+1 ) );
294 if(iFormattedNumber->Length())
296 CPNGNumberGrouping* pThis = const_cast<CPNGNumberGrouping*>(this);
297 pThis->doClearFormattedNumbers();
300 iFormattedNumberPtr.Set(KNullDesC);
303 return iFormattedNumberPtr;
306 EXPORT_C const TDesC& CPNGNumberGrouping::FormattedNumber() const
308 if( !iFormattedNumber->Length() )
312 CPNGNumberGrouping* pThis = const_cast<CPNGNumberGrouping*>(this);
314 if( LengthToGroup() < KMinimumLengthToGroup || !iExtension->iNumberGroupingCRValue )
316 // This is now just a short cut, as doNumberGroupingL handles premature truncation of
317 // formatting. But this avoids all the language checking
318 doNumberSquashing(); // copies the unformatted number straight into the formatted number
323 if(iForceLanguage != ELangTest)
324 eLanguage = iForceLanguage;
326 eLanguage = doReadLanguageFromSharedData();
328 if(eLanguage != iLanguage)
330 iLanguage = eLanguage;
332 TRAP(err, pThis->doReadFormatInfoFromResourceFileL());
335 iFormattedNumberPtr.Set(KNullDesC);
336 return iFormattedNumberPtr;
340 TRAP(err, doNumberGroupingL());
344 pThis->doClearFormattedNumbers();
346 iFormattedNumberPtr.Set(iFormattedNumber->Ptr(), iFormattedNumber->Length());
349 return iFormattedNumberPtr;
352 EXPORT_C const TDesC& CPNGNumberGrouping::ReverseFormattedNumber(TInt aFrom, TInt aTo) const
356 if(iUnformattedNumber->Length() != 0 &&
360 ReverseFormattedNumber();
362 iReverseFormattedNumberPtr.Set(KNullDesC);
364 TInt length = iReverseFormattedNumber->Length();
365 if( aTo < length + 1 )
367 // Advance to the next non-space
368 if( iExtension->iNumberGroupingCRValue )
370 while( (aFrom < length ) && (*iReverseFormattedNumber)[aFrom] == KNumberGroupingSpace )
375 // Retreat to the last non-space
376 while( (aTo > 0) && (*iReverseFormattedNumber)[aTo] == KNumberGroupingSpace )
382 // Does fetching the descriptor still make sense?
383 if ( (0 <= aFrom) && (aFrom <= aTo) && (aTo < length) )
384 iReverseFormattedNumberPtr.Set(
385 iReverseFormattedNumber->Mid( aFrom, aTo-aFrom+1) );
389 iReverseFormattedNumberPtr.Set(KNullDesC);
392 return iReverseFormattedNumberPtr; // Zero initialized at construction
395 EXPORT_C const TDesC& CPNGNumberGrouping::ReverseFormattedNumber() const
397 if( iReverseFormattedNumber && !iReverseFormattedNumber->Length())
399 if(!iFormattedNumber->Length())
402 TInt nLength = iFormattedNumber->Length();
404 TPtr ptrModifyable(iReverseFormattedNumber->Des());
407 for(TInt i = nLength; i > 0; --i)
409 TText cChar = (*iFormattedNumber)[i-1];
411 ptrModifyable.Insert(nLength - i, bufChar);
414 iReverseFormattedNumberPtr.Set(iReverseFormattedNumber->Ptr(), nLength);
417 return iReverseFormattedNumberPtr;
420 EXPORT_C const TDesC& CPNGNumberGrouping::Selection(TInt aFrom, TInt aTo) const
422 if(aFrom < iUnformattedNumber->Length())
424 TPtr ptrUnformatted = iUnformattedNumber->Des();
425 iSelectionPtr.Set(&(ptrUnformatted[aFrom]), aTo - aFrom);
428 iSelectionPtr.Set(KNullDesC);
430 return iSelectionPtr;
433 EXPORT_C const TDesC& CPNGNumberGrouping::UnFormattedNumber(TInt aFrom, TInt aTo) const
435 if (iUnformattedNumber && aFrom >= 0 && aFrom <= aTo && aTo < iUnformattedNumber->Length())
437 iUnformattedNumberPtr.Set(&((*iUnformattedNumber)[aFrom]), aTo - aFrom + 1);
441 iUnformattedNumberPtr.Set(KNullDesC);
443 return iUnformattedNumberPtr;
446 EXPORT_C const TDesC& CPNGNumberGrouping::UnFormattedNumber() const
448 return UnFormattedNumber(0, iUnformattedNumber->Length() - 1);
451 TLanguage CPNGNumberGrouping::doReadLanguageFromSharedData() const
453 if (iExtension->iNumberGroupingCRValue)
455 return ELangAmerican;
463 void CPNGNumberGrouping::doClearFormattedNumbers()
465 TPtr ptrModifyable( iUnformattedNumber->Des() );
467 for (TInt index = 0; index < ptrModifyable.Length(); index++)
469 TChar ch = TChar(ptrModifyable[index]);
470 ch.Fold( TChar::EFoldDigits | TChar::EFoldSpaces);
473 iFormattedNumber->Des().Zero();
474 iFormattedNumberPtr.Set(KNullDesC);
476 if ( iReverseFormattedNumber )
477 iReverseFormattedNumber->Des().Zero();
479 iReverseFormattedNumberPtr.Set(KNullDesC);
480 iMatchedPatternIndex = ENoMatchedPattern;
483 void CPNGNumberGrouping::doReadFormatInfoFromResourceFileL()
485 doClearGroupingItemsList();
489 RPointerArray<TDesC> parrGroupingPatternsList;
490 CleanupResetAndDestroyPushL(parrGroupingPatternsList);
492 TInt maxExtraCharacters(0);
495 CleanupClosePushL(fs);
496 if(fs.Connect() == KErrNone)
498 RResourceFile resourceFile;
499 CleanupClosePushL(resourceFile);
501 resourceFile.OpenL(fs, _L("z:\\resource\\numbergrouping.rsc"));
502 HBufC8* bufResource = resourceFile.AllocReadL(R_GROUPING_MAPPING);
504 TResourceReader resourceReader;
505 resourceReader.SetBuffer(bufResource);
507 TInt nLanguageCount = resourceReader.ReadInt8();
508 TBool bLanguageMatches = EFalse;
510 while(nLanguageCount-- || !bLanguageMatches)
512 TBool bLanguageMatches = (resourceReader.ReadInt8() == iLanguage);
514 if(bLanguageMatches || ((nLanguageCount == -1) && !bLanguageMatches))
516 TInt nGroupingSchemeCount = resourceReader.ReadInt8();
518 while(nGroupingSchemeCount--)
520 TInt thisMaxExtraCharacters(0);
522 resourceReader, parrGroupingPatternsList, thisMaxExtraCharacters );
523 // take this new max extra characters if bigger
524 maxExtraCharacters = Max( maxExtraCharacters, thisMaxExtraCharacters );
527 break; // This breaks out because we take the first language that matches
529 } // End of if on language/locale test
530 else // skip other locales
532 TInt nGroupingSchemeCount = resourceReader.ReadInt8();
533 while(nGroupingSchemeCount--)
535 SkipGroupingSchemeL( resourceReader );
542 resourceFile.Close();
543 CleanupStack::Pop(); // resource file
547 CleanupStack::Pop(); // file system
549 iExtension->iMaxExtraCharacters = maxExtraCharacters; // Latch the high water mark of extra characters
551 iRegExp = CRegularExpression::NewL(&parrGroupingPatternsList);
553 CleanupStack::PopAndDestroy(&parrGroupingPatternsList); // patterns list
556 void CPNGNumberGrouping::doNumberGroupingL() const
558 TInt lengthToGroup = LengthToGroup();
560 if ( lengthToGroup >= KMinimumLengthToGroup )
563 TInt matchedPattern = KErrNotFound;
564 TInt newMatchedPattern = KErrNotFound;
566 // Search for matches in the RegExp object. It returns the next matching pattern
567 // However, even if there is a match, lengthToGroup may not be in the deployment
568 // length range between minDigits and MaxDigits, inclusive
570 // Check for another matching pattern
571 newMatchedPattern = iRegExp->SearchFrom( newMatchedPattern+1, *iUnformattedNumber);
573 if( newMatchedPattern != KErrNotFound) // Found a match, but it is OK?
576 TInt minDigits = iGroupingItemsList[newMatchedPattern]->iMinNumberOfDigits;
577 TInt maxDigits = iGroupingItemsList[newMatchedPattern]->iMaxNumberOfDigits;
579 // Fill in sensible values for min and max if not present
583 maxDigits = lengthToGroup;
585 if ( minDigits <= lengthToGroup && lengthToGroup <= maxDigits )
587 matchedPattern = newMatchedPattern; // accept this new pattern
592 } while ( newMatchedPattern != KErrNotFound );
594 // Actually go and do the grouping
595 if ( matchedPattern != KErrNotFound )
597 doNumberGroupingForPatternL( matchedPattern, lengthToGroup );
603 // if we get to here, either the string was not matched to any of the patterns or the
604 // unformatted string is exactly the display length. In either case we call
605 // doNumberSquashing() which simply leaves the string as it is...
611 void CPNGNumberGrouping::doNumberGroupingForPatternL( TInt aMatchingPattern, TInt aLengthToGroup ) const
613 iMatchedPatternIndex = aMatchingPattern;
618 TPtr desUnformattedNumber = iUnformattedNumber->Des();
619 TInt unformattedLength = iUnformattedNumber->Length();
621 __ASSERT_ALWAYS( aLengthToGroup <= unformattedLength , Panic(ENumberGroupingBadLengthToGroup) );
623 TPNGGroupingInfo* matchedPattern = iGroupingItemsList[iMatchedPatternIndex];
624 TInt nAfterCount = matchedPattern->iAfterPositions.Count();
625 TBool bBeforePosition = (matchedPattern->iBeforePosition.iPosition == -1)?0:1;
627 // Test to see if the beforePosition can be used with the current text length.
628 // The following does not allow the before position to be used if it would result in an
629 // insertion right next to one from the AfterPositions.
630 // That is, tildas in the formatting string represent 1 or more characters.
631 // e.g. if the last afterPosition is 4 and the before position is 3, then a 7 digit
632 // number will not be able to have the before position used.
634 (unformattedLength - matchedPattern->iBeforePosition.iPosition) <=
635 matchedPattern->iAfterPositions[nAfterCount - 1].iPosition)
637 bBeforePosition = EFalse;
640 TPtr ptrModifyable(iFormattedNumber->Des());
642 for(TInt i = 0; i < nAfterCount && nHighPos < aLengthToGroup ; ++i)
644 nHighPos = matchedPattern->iAfterPositions[i].iPosition;
645 if ( nHighPos >= aLengthToGroup )
648 if(nHighPos < unformattedLength)
650 ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
651 ptrModifyable.Append(matchedPattern->iAfterPositions[i].iSeparatorCharacter);
656 // Do not do "before end" formatting at all if there is any truncation
657 if ( aLengthToGroup < unformattedLength )
659 TInt nBeforePosition = matchedPattern->iBeforePosition.iPosition;
661 if(bBeforePosition && nBeforePosition < unformattedLength)
663 nHighPos = unformattedLength - nBeforePosition;
664 ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
665 ptrModifyable.Append( matchedPattern->iBeforePosition.iSeparatorCharacter );
670 nHighPos = unformattedLength;
671 ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
675 void CPNGNumberGrouping::doNumberSquashing() const
677 __ASSERT_ALWAYS( !iFormattedNumber->Length(), Panic(ENumberGroupingFormattedNumberAlreadyExists) );
679 // just copy from one t'other...
680 TPtr ptrModifyable(iFormattedNumber->Des());
681 ptrModifyable.Copy(*iUnformattedNumber);
682 iMatchedPatternIndex = ENoMatchedPattern;
685 void CPNGNumberGrouping::doClearGroupingItemsList()
687 TInt nCount = iGroupingItemsList.Count();
689 for(TInt i = 0; i < nCount; ++i)
691 iGroupingItemsList[i]->iAfterPositions.Close();
692 delete iGroupingItemsList[i];
693 iGroupingItemsList[i] = NULL;
695 iGroupingItemsList.Close();
698 void CPNGNumberGrouping::ReadGroupingSchemeL(
699 TResourceReader& aResourceReader,
700 RPointerArray<TDesC>& aGroupingPatternsList,
701 TInt& aMaxExtraCharacters )
703 CleanupResetAndDestroyPushL(aGroupingPatternsList);
704 TPNGGroupingInfo* groupingInfo = new (ELeave) TPNGGroupingInfo;
705 CleanupStack::PushL( groupingInfo );
707 // Read in all resource for this grouping scheme, perform checking and then analyze it
708 HBufC* initialDigits = aResourceReader.ReadHBufCL();
709 __ASSERT_ALWAYS( initialDigits, Panic( ENumberGroupingNoInitialDigitsInResource ) );
710 CleanupStack::PushL( initialDigits );
712 groupingInfo->iMinNumberOfDigits = aResourceReader.ReadInt8();
713 groupingInfo->iMaxNumberOfDigits = aResourceReader.ReadInt8();
715 ( groupingInfo->iMaxNumberOfDigits == -1) ||
716 ( groupingInfo->iMinNumberOfDigits <= groupingInfo->iMaxNumberOfDigits ),
717 Panic( ENumberGroupingBadMinMaxDigitRangeInResource ) );
719 // Read in formatting Pattern
720 HBufC* formatPattern = aResourceReader.ReadHBufCL();
722 if ( formatPattern ) // Does not have to be there
724 CleanupStack::PushL( formatPattern );
725 TInt formatLength = formatPattern->Length();
726 if ( formatLength > 0 )
728 // Obtain a wildcard version of the matching pattern in initialDigits.
729 // This is used to check the supplied formatPattern for comformance to initialDigits
730 HBufC* wildcardedMatchBuf = HBufC::NewLC( formatLength ); // Will not be longer than the search pattern
732 TPtr wildcardedMatchPtr( wildcardedMatchBuf->Des() );
733 // Get the example number using the latest search pattern only
734 GetWildcardVersionOfMatchStringL( *initialDigits, KNumberGroupingWildcard, wildcardedMatchPtr );
736 // Now parse the descriptor
737 TBool trailingPossible(EFalse);
738 ParseForAfterPositions(
739 *formatPattern, groupingInfo, wildcardedMatchPtr, aMaxExtraCharacters, trailingPossible );
741 // Now parse the descriptor from the end if needed
742 if ( trailingPossible )
743 ParseForBeforePosition( *formatPattern, groupingInfo, aMaxExtraCharacters );
745 CleanupStack::PopAndDestroy( wildcardedMatchBuf );
747 CleanupStack::PopAndDestroy( formatPattern );
748 } // End of if on formatPattern.Length
750 User::LeaveIfError( aGroupingPatternsList.Append( initialDigits ) );
751 CleanupStack::Pop( initialDigits );
753 // Do not leave if the next one fails, but remove the last from the patterns list and then leave
754 // This is done in case someone TRAPs. Otherwise neither of these lists would be used and their
755 // mismatch would not be a problem
756 if ( TInt err = iGroupingItemsList.Append(groupingInfo) != KErrNone )
758 // return value of Count will be at least 1, because we have just successfully gone through an Append
759 aGroupingPatternsList.Remove( aGroupingPatternsList.Count() - 1 );
760 // ownership is now mine again...
761 delete initialDigits;
762 // Need to delete groupingInfo, and make sure it is no longer on the cleanupstack
763 CleanupStack::PopAndDestroy( groupingInfo );
768 CleanupStack::Pop( groupingInfo ); // Success. This object now not owned by the cleanupstack
771 CleanupStack::Pop(&aGroupingPatternsList);
774 void CPNGNumberGrouping::ParseForAfterPositions(
775 const TDesC& aFormatPattern,
776 TPNGGroupingInfo* aGroupingInfo,
777 const TDesC& aWildcardedMatchingPattern,
778 TInt& aMaxExtraCharacters,
779 TBool& trailingPossible ) const
781 TInt pos(0); // Keeps track of the position with which the next separator will be stored
782 TInt formatLength = aFormatPattern.Length();
783 for (TInt index = 0; index < formatLength; index++ )
785 // The format pattern is compared with the matching pattern. The matching pattern may be
786 // shorter than the format pattern, so by default a wildcard character is used.
787 TText ch = aFormatPattern[index];
788 TText matchingChar(KNumberGroupingWildcard); // default to expect is the wildcard character
789 if ( pos < aWildcardedMatchingPattern.Length() ) // if still within the matching pattern
790 matchingChar = aWildcardedMatchingPattern[pos];
791 if ( ch == matchingChar )
792 pos++; // not a separator. index where the next "after" marker goes
793 else if ( ch == KNumberGroupingOneOrMoreCharactersToken )
795 // finish looking for "afterPositions". But there may be a "before" position in the
796 // remainder, so set the flag
797 trailingPossible = ETrue;
802 // Explicit prevention of any separator characters being valid phone numbers
804 if ( IsValidPhoneNumberCharacter( ch ) || ch == KNumberGroupingWildcard )
807 _L("NumberGrouping: Illegal character or format mismatch in resource: initialDigits pattern= <%S> formatPattern=<%S>"),
808 &aWildcardedMatchingPattern, &aFormatPattern );
811 __ASSERT_DEBUG( !IsValidPhoneNumberCharacter( ch ), Panic( ENumberGroupingInvalidSeparatorCharacterInFormat ) );
812 __ASSERT_DEBUG( ch != KNumberGroupingWildcard, Panic( ENumberGroupingMatchingPatternVersusFormatPatternMismatch ) );
813 TPNGSeparator separator( pos, aFormatPattern[index]);
814 aGroupingInfo->iAfterPositions.Append(separator);
815 aMaxExtraCharacters++;
820 void CPNGNumberGrouping::ParseForBeforePosition(
821 const TDesC& aFormatPattern,
822 TPNGGroupingInfo* aGroupingInfo,
823 TInt& aMaxExtraCharacters ) const
826 TInt formatLength = aFormatPattern.Length();
828 for (TInt index = formatLength-1; index >=0; index-- )
830 TText ch = aFormatPattern[index];
831 if ( ch == KNumberGroupingWildcard )
833 else if ( ch == KNumberGroupingOneOrMoreCharactersToken )
837 // Explicit prevention of any separator characters being valid phone numbers
839 if ( IsValidPhoneNumberCharacter( ch ) )
842 _L("NumberGrouping: Illegal character in trailing part of format string in resource: formatPattern=<%S>"),
846 __ASSERT_DEBUG( !IsValidPhoneNumberCharacter( ch ),
847 Panic( ENumberGroupingInvalidSeparatorCharacterInFormat ) );
848 TPNGSeparator separator( pos, ch );
849 aGroupingInfo->iBeforePosition = separator;
850 aMaxExtraCharacters++;
857 void CPNGNumberGrouping::SkipGroupingSchemeL( TResourceReader& aResourceReader ) const
860 tempBuf = aResourceReader.ReadHBufCL();
862 aResourceReader.Advance(2); // min and max characters
863 tempBuf = aResourceReader.ReadHBufCL();
867 void CPNGNumberGrouping::GetWildcardVersionOfMatchStringL(
868 const TDesC& aMatchString,
870 TDes& aWildcardMatchString ) const
872 RPointerArray<TDesC> patternList;
873 CleanupClosePushL(patternList);
875 // Make a copy of the input string
876 HBufC* matchString = aMatchString.AllocLC();
878 User::LeaveIfError( patternList.Append(matchString) );// takes ownership
879 CleanupStack::Pop( matchString );
881 CRegularExpression* regExp = CRegularExpression::NewLC(&patternList);
883 // Only 1 pattern fed in. Access that pattern at index 0
884 regExp->GetWildcardVersionOfPattern( 0 , aWildcard, aWildcardMatchString );
886 CleanupStack::PopAndDestroy(regExp);
888 // Delete the patterns list
889 delete patternList[0];
890 CleanupStack::PopAndDestroy();
894 EXPORT_C TBool CPNGNumberGrouping::IsCharacterInsertedByNumberGrouping(TInt aPos) const
896 TInt insertedCharacters = Length() - UnFormattedLength();
898 if( insertedCharacters == 0 ) // no formatting was done
900 else if ( aPos < ( insertedCharacters + LengthToGroup() ) )
902 return !IsValidPhoneNumberCharacter( (*iFormattedNumber)[aPos] );
904 else // aPos is pointing at or beyond index= LengthToGroup() + <chars inserted>; no formatting there
909 TBool CPNGNumberGrouping::IsValidPhoneNumberCharacter( TText aCharacter ) const
911 if ( ((TChar)aCharacter).IsDigit() )
914 // Check through the list of additional valid phone number characters
915 TInt numAdditionalChars = sizeof( KAdditionalPhoneNumberCharacters )/sizeof(TText);
917 for (TInt index = 0; index < numAdditionalChars; index++)
919 if ( aCharacter == KAdditionalPhoneNumberCharacters[index] )
926 EXPORT_C TBool CPNGNumberGrouping::IsChangedByGrouping() const
928 // The only way that grouping is effectively different is by making things longer
929 return ( Length() > UnFormattedLength() );
932 TInt CPNGNumberGrouping::LengthToGroup() const
935 TPtrC ptr = iUnformattedNumber->Des();
936 TInt lengthToGroup = ptr.Length();
938 // Find the first non-digit
939 for (TInt index = 0; index < ptr.Length(); index++)
941 TChar ch = TChar(ptr[index]);
942 ch.Fold(TChar::EFoldDigits);
943 if ( !( ch.IsDigit() ) )
945 lengthToGroup = index; // only characters BEFORE the character at index are grouped
950 return lengthToGroup;