os/textandloc/textandlocutils/numbergrouping/src/NumberGrouping.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 2002-2008 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of the License "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
#include <f32file.h>
sl@0
    19
#include <barsc.h> 
sl@0
    20
#include "NumberGrouping.h"
sl@0
    21
#include "RegularExpression.h"
sl@0
    22
#include "cleanuputil.h"
sl@0
    23
sl@0
    24
#include <barsread.h>
sl@0
    25
//#include <eikenv.h>
sl@0
    26
#include <centralrepository.h>
sl@0
    27
#include <NumberGroupingCRKeys.h>
sl@0
    28
sl@0
    29
const TText KNumberGroupingWildcard('n');
sl@0
    30
const TText KNumberGroupingOneOrMoreCharactersToken('~');
sl@0
    31
sl@0
    32
// This constant represents all the valid whitespace that we can handle. Several
sl@0
    33
// APIs give whitespace special significance as a formatting character, especially
sl@0
    34
// those methods used to obtain text for formatting into fixed-width UI elements
sl@0
    35
// where spaces are not to be rendered at the margins.
sl@0
    36
const TText KNumberGroupingSpace(' ');
sl@0
    37
sl@0
    38
const TInt KMinimumLengthToGroup = 1; // No grouping occurs if fewer than this in unformatted buffer
sl@0
    39
sl@0
    40
#include <numbergrouping.rsg>
sl@0
    41
sl@0
    42
sl@0
    43
GLDEF_C void Panic(TNumberGroupingPanic aPanic)
sl@0
    44
    {
sl@0
    45
    _LIT(KPanicCat,"Number Grouping");
sl@0
    46
    User::Panic(KPanicCat, aPanic);
sl@0
    47
    }
sl@0
    48
sl@0
    49
// Valid phone number characters apart from IsDigit() characters are listed here
sl@0
    50
const TText KAdditionalPhoneNumberCharacters[] = {'+','#','*','p','w'};
sl@0
    51
sl@0
    52
sl@0
    53
sl@0
    54
sl@0
    55
NONSHARABLE_CLASS(CPNGNumberGroupingExtension): public CBase
sl@0
    56
	{
sl@0
    57
public:
sl@0
    58
	CPNGNumberGroupingExtension();
sl@0
    59
	~CPNGNumberGroupingExtension();
sl@0
    60
public:
sl@0
    61
    TInt        iMaxExtraCharacters; 
sl@0
    62
    TInt        iNumberGroupingCRValue;
sl@0
    63
    };
sl@0
    64
sl@0
    65
CPNGNumberGroupingExtension::CPNGNumberGroupingExtension()
sl@0
    66
	{
sl@0
    67
	}
sl@0
    68
sl@0
    69
CPNGNumberGroupingExtension::~CPNGNumberGroupingExtension()
sl@0
    70
	{
sl@0
    71
	}
sl@0
    72
sl@0
    73
sl@0
    74
sl@0
    75
CPNGNumberGrouping::TPNGSeparator::TPNGSeparator()
sl@0
    76
    : iPosition(-1), iSeparatorCharacter(KNumberGroupingSpace)
sl@0
    77
    {
sl@0
    78
    }
sl@0
    79
sl@0
    80
CPNGNumberGrouping::TPNGSeparator::TPNGSeparator( TInt aPosition, TText aSeparatorCharacter )
sl@0
    81
    : iPosition(aPosition), iSeparatorCharacter(aSeparatorCharacter)
sl@0
    82
    {
sl@0
    83
    }
sl@0
    84
sl@0
    85
CPNGNumberGrouping::TPNGGroupingInfo::TPNGGroupingInfo()
sl@0
    86
    {
sl@0
    87
    }
sl@0
    88
sl@0
    89
// CPNGNumberGrouping - grouping engine class
sl@0
    90
CPNGNumberGrouping::CPNGNumberGrouping(TInt aMaxLength /* = 0 */, TBool aReversed /* = EFalse */) :
sl@0
    91
                                            iForceLanguage(ELangTest),
sl@0
    92
                                            iUnformattedNumberPtr(KNullDesC),
sl@0
    93
                                            iFormattedNumberPtr(KNullDesC),
sl@0
    94
                                            iReverseFormattedNumberPtr(KNullDesC),
sl@0
    95
                                            iSelectionPtr(KNullDesC),
sl@0
    96
                                            iLanguage(ELangTest),
sl@0
    97
                                            iMaxUnformattedLength(aMaxLength),
sl@0
    98
                                            iReversed(aReversed),
sl@0
    99
                                            iMatchedPatternIndex(ENoMatchedPattern)
sl@0
   100
    {
sl@0
   101
    }
sl@0
   102
sl@0
   103
EXPORT_C CPNGNumberGrouping* CPNGNumberGrouping::NewL( TInt aMaxLength, TBool aReserved)
sl@0
   104
    {
sl@0
   105
    CPNGNumberGrouping* s = NewLC(aMaxLength, aReserved);
sl@0
   106
    CleanupStack::Pop();
sl@0
   107
    return s;
sl@0
   108
    }
sl@0
   109
sl@0
   110
EXPORT_C CPNGNumberGrouping* CPNGNumberGrouping::NewLC( TInt aMaxLength, TBool aReserved)
sl@0
   111
    {
sl@0
   112
    CPNGNumberGrouping* s = new(ELeave)CPNGNumberGrouping( aMaxLength, aReserved);
sl@0
   113
    CleanupStack::PushL(s);
sl@0
   114
    s->ConstructL();
sl@0
   115
    return s;
sl@0
   116
    }
sl@0
   117
sl@0
   118
void CPNGNumberGrouping::ConstructL()
sl@0
   119
    {
sl@0
   120
    iExtension = new (ELeave) CPNGNumberGroupingExtension(); 
sl@0
   121
    CRepository* repository = NULL;
sl@0
   122
    iExtension->iNumberGroupingCRValue = 0;
sl@0
   123
    TRAPD(ret, repository = CRepository::NewL(KCRUidNumberGrouping));
sl@0
   124
    if (ret == KErrNone)
sl@0
   125
        {
sl@0
   126
        ret = repository->Get(KNumberGrouping, iExtension->iNumberGroupingCRValue);
sl@0
   127
        }
sl@0
   128
    delete repository;
sl@0
   129
    
sl@0
   130
    // Read from resource first in order to obtain a value for iExtension->iMaxExtraCharacters
sl@0
   131
    iLanguage = doReadLanguageFromSharedData();
sl@0
   132
    doReadFormatInfoFromResourceFileL(); // This sets iExtension->iMaxExtraCharacters
sl@0
   133
sl@0
   134
    // Allocation of buffers.  Note that iMaxUnformattedLength must be retained as member
sl@0
   135
    // data because HBufCs may come back with more storage available than asked for.
sl@0
   136
    // The uncertainty in the actual length provided by the HBufCs means that although
sl@0
   137
    // they are asked to be different by iExtension->iMaxExtraCharacters, the difference between the
sl@0
   138
    // actual MaxLengths may be less than iMaxExtraCharcaters.
sl@0
   139
    // The approach decided upon is to retain iMaxUnformattedLength as member data and
sl@0
   140
    // use IT everywhere throughout this class's implementation, NEVER using
sl@0
   141
    // iUnformattedNumber->Des().MaxLength()
sl@0
   142
    iUnformattedNumber = HBufC::NewL(iMaxUnformattedLength);
sl@0
   143
    iFormattedNumber = HBufC::NewL(iMaxUnformattedLength + iExtension->iMaxExtraCharacters);
sl@0
   144
sl@0
   145
    // Create revesed buffer only if requested
sl@0
   146
    if ( iReversed )
sl@0
   147
        iReverseFormattedNumber = HBufC::NewL(iMaxUnformattedLength + iExtension->iMaxExtraCharacters);        
sl@0
   148
    }
sl@0
   149
sl@0
   150
EXPORT_C CPNGNumberGrouping::~CPNGNumberGrouping()
sl@0
   151
    {
sl@0
   152
    doClearGroupingItemsList();
sl@0
   153
sl@0
   154
    delete iUnformattedNumber;
sl@0
   155
    delete iFormattedNumber;
sl@0
   156
    delete iReverseFormattedNumber;
sl@0
   157
    delete iRegExp;
sl@0
   158
    delete iExtension;
sl@0
   159
    }
sl@0
   160
sl@0
   161
EXPORT_C TInt CPNGNumberGrouping::Insert(TInt aIndex, TText aChar)
sl@0
   162
    {
sl@0
   163
sl@0
   164
    if( aIndex >= 0 && aIndex <= iUnformattedNumber->Length())
sl@0
   165
        {
sl@0
   166
sl@0
   167
        if(iUnformattedNumber->Length() >= iMaxUnformattedLength)
sl@0
   168
            return KErrOverflow;
sl@0
   169
sl@0
   170
        TBuf<1> bufChar(1);
sl@0
   171
        bufChar[0] = aChar;
sl@0
   172
        TPtr ptrModifyable(iUnformattedNumber->Des());
sl@0
   173
        ptrModifyable.Insert(aIndex, bufChar);
sl@0
   174
sl@0
   175
        doClearFormattedNumbers();
sl@0
   176
sl@0
   177
        return KErrNone;
sl@0
   178
        }
sl@0
   179
sl@0
   180
    return KErrIndexOutOfRange;
sl@0
   181
    }
sl@0
   182
sl@0
   183
EXPORT_C TInt CPNGNumberGrouping::Delete(TInt aIndex)
sl@0
   184
    {
sl@0
   185
    if(aIndex >= 0 && aIndex < iUnformattedNumber->Length())
sl@0
   186
        {
sl@0
   187
        TPtr ptrModifyable(iUnformattedNumber->Des());
sl@0
   188
        ptrModifyable.Delete(aIndex, KSingleCharacter);
sl@0
   189
sl@0
   190
        doClearFormattedNumbers();
sl@0
   191
sl@0
   192
        return KErrNone;
sl@0
   193
        }
sl@0
   194
sl@0
   195
    return KErrIndexOutOfRange;
sl@0
   196
    }
sl@0
   197
sl@0
   198
EXPORT_C TInt CPNGNumberGrouping::Append(TText aChar)
sl@0
   199
    {
sl@0
   200
    if(iUnformattedNumber->Length() >= iMaxUnformattedLength)
sl@0
   201
        return KErrOverflow;
sl@0
   202
sl@0
   203
    TBuf<1> bufChar(1);
sl@0
   204
    bufChar[0] = aChar;
sl@0
   205
    TPtr ptrModifyable(iUnformattedNumber->Des());
sl@0
   206
    ptrModifyable.Append(bufChar);
sl@0
   207
sl@0
   208
    doClearFormattedNumbers();
sl@0
   209
sl@0
   210
    return KErrNone;
sl@0
   211
    }
sl@0
   212
sl@0
   213
EXPORT_C TInt CPNGNumberGrouping::Set(const TDesC& aNumber)
sl@0
   214
    {
sl@0
   215
    if( aNumber.Length() > iMaxUnformattedLength )
sl@0
   216
        return KErrOverflow;
sl@0
   217
sl@0
   218
    TPtr ptrModifyable( iUnformattedNumber->Des() );
sl@0
   219
    ptrModifyable.Copy( aNumber );
sl@0
   220
sl@0
   221
    doClearFormattedNumbers();
sl@0
   222
sl@0
   223
    return KErrNone;
sl@0
   224
    }
sl@0
   225
sl@0
   226
EXPORT_C TInt CPNGNumberGrouping::Length() const
sl@0
   227
    {
sl@0
   228
    if(!iFormattedNumber->Length()) // This test is used as the trigger to reformat
sl@0
   229
        FormattedNumber();
sl@0
   230
sl@0
   231
    return iFormattedNumber->Length();
sl@0
   232
    }
sl@0
   233
sl@0
   234
EXPORT_C TInt CPNGNumberGrouping::UnFormattedLength() const
sl@0
   235
    {
sl@0
   236
    return iUnformattedNumber->Length();
sl@0
   237
    }
sl@0
   238
sl@0
   239
EXPORT_C TInt CPNGNumberGrouping::MaxDisplayLength() const
sl@0
   240
    {
sl@0
   241
    // Despite its name, this method returns the max length of the UNFORMATTED buffer
sl@0
   242
    // This must not be implemented to return the actual length available in
sl@0
   243
    // iUnformattedBuffer.
sl@0
   244
    return iMaxUnformattedLength;
sl@0
   245
    }
sl@0
   246
sl@0
   247
EXPORT_C TBool CPNGNumberGrouping::IsSpace(TInt aPos) const
sl@0
   248
    {
sl@0
   249
    // Very tricky semantics for this.  Must be a space inserted by the formatting
sl@0
   250
    if ( iFormattedNumber->Length() > aPos &&
sl@0
   251
         iFormattedNumber->operator[](aPos) == KNumberGroupingSpace &&
sl@0
   252
         // Check also that is is less than the length to group + inserted characters
sl@0
   253
         aPos < ( LengthToGroup() + (iFormattedNumber->Length() - iUnformattedNumber->Length()) ) )
sl@0
   254
        return ETrue;
sl@0
   255
    else
sl@0
   256
        return EFalse;
sl@0
   257
    }
sl@0
   258
sl@0
   259
EXPORT_C const TDesC& CPNGNumberGrouping::FormattedNumber(TInt aFrom, TInt aTo) const
sl@0
   260
    {
sl@0
   261
    if(iUnformattedNumber->Length() != 0 &&
sl@0
   262
        aFrom >= 0 &&
sl@0
   263
        aFrom <= aTo )
sl@0
   264
        {
sl@0
   265
        FormattedNumber();
sl@0
   266
        iFormattedNumberPtr.Set(KNullDesC);
sl@0
   267
sl@0
   268
        TInt length = iFormattedNumber->Length();
sl@0
   269
        if(aTo < length + 1)
sl@0
   270
            {
sl@0
   271
            TInt length = iFormattedNumber->Length();
sl@0
   272
            if ( iExtension->iNumberGroupingCRValue )
sl@0
   273
            	{
sl@0
   274
				// Advance to the next non-space
sl@0
   275
				while( (aFrom < length ) && (*iFormattedNumber)[aFrom] == KNumberGroupingSpace )
sl@0
   276
					{
sl@0
   277
					++aFrom;
sl@0
   278
					}
sl@0
   279
	
sl@0
   280
				// Retreat to the last non-space
sl@0
   281
				while( (aTo > 0) && (*iFormattedNumber)[aTo] == KNumberGroupingSpace )
sl@0
   282
					{
sl@0
   283
					--aTo;
sl@0
   284
					}
sl@0
   285
            	}
sl@0
   286
sl@0
   287
            // Does fetching the descriptor still make sense?
sl@0
   288
            if ( (0 <= aFrom) && (aFrom <= aTo) && (aTo < length) )
sl@0
   289
                iFormattedNumberPtr.Set( iFormattedNumber->Mid( aFrom, aTo-aFrom+1 ) );
sl@0
   290
            }
sl@0
   291
        }
sl@0
   292
    else
sl@0
   293
        {
sl@0
   294
        if(iFormattedNumber->Length())
sl@0
   295
            {
sl@0
   296
            CPNGNumberGrouping* pThis = const_cast<CPNGNumberGrouping*>(this);
sl@0
   297
            pThis->doClearFormattedNumbers();
sl@0
   298
            }
sl@0
   299
sl@0
   300
        iFormattedNumberPtr.Set(KNullDesC);
sl@0
   301
        }
sl@0
   302
sl@0
   303
    return iFormattedNumberPtr;
sl@0
   304
    }
sl@0
   305
sl@0
   306
EXPORT_C const TDesC& CPNGNumberGrouping::FormattedNumber() const
sl@0
   307
    {
sl@0
   308
    if( !iFormattedNumber->Length() )
sl@0
   309
        {
sl@0
   310
        TInt err = KErrNone;
sl@0
   311
sl@0
   312
        CPNGNumberGrouping* pThis = const_cast<CPNGNumberGrouping*>(this);
sl@0
   313
sl@0
   314
        if( LengthToGroup() < KMinimumLengthToGroup || !iExtension->iNumberGroupingCRValue )
sl@0
   315
            {
sl@0
   316
            // This is now just a short cut, as doNumberGroupingL handles premature truncation of
sl@0
   317
            // formatting. But this avoids all the language checking
sl@0
   318
            doNumberSquashing();  // copies the unformatted number straight into the formatted number
sl@0
   319
            }
sl@0
   320
        else
sl@0
   321
            {
sl@0
   322
            TLanguage eLanguage;
sl@0
   323
            if(iForceLanguage != ELangTest)
sl@0
   324
                eLanguage = iForceLanguage;
sl@0
   325
            else
sl@0
   326
                eLanguage = doReadLanguageFromSharedData();
sl@0
   327
sl@0
   328
            if(eLanguage != iLanguage)
sl@0
   329
                {
sl@0
   330
                iLanguage = eLanguage;
sl@0
   331
sl@0
   332
                TRAP(err, pThis->doReadFormatInfoFromResourceFileL());
sl@0
   333
                if(err != KErrNone)
sl@0
   334
                    {
sl@0
   335
                    iFormattedNumberPtr.Set(KNullDesC);
sl@0
   336
                    return iFormattedNumberPtr;
sl@0
   337
                    }
sl@0
   338
                }
sl@0
   339
sl@0
   340
            TRAP(err, doNumberGroupingL());
sl@0
   341
            }
sl@0
   342
sl@0
   343
        if(err != KErrNone)
sl@0
   344
            pThis->doClearFormattedNumbers();
sl@0
   345
        else
sl@0
   346
            iFormattedNumberPtr.Set(iFormattedNumber->Ptr(), iFormattedNumber->Length());
sl@0
   347
        }
sl@0
   348
sl@0
   349
    return iFormattedNumberPtr;
sl@0
   350
    }
sl@0
   351
sl@0
   352
EXPORT_C const TDesC& CPNGNumberGrouping::ReverseFormattedNumber(TInt aFrom, TInt aTo) const
sl@0
   353
    {
sl@0
   354
    if ( iReversed )
sl@0
   355
        {
sl@0
   356
        if(iUnformattedNumber->Length() != 0 &&
sl@0
   357
            aFrom >= 0 &&
sl@0
   358
            aFrom <= aTo)
sl@0
   359
            {
sl@0
   360
            ReverseFormattedNumber();
sl@0
   361
sl@0
   362
            iReverseFormattedNumberPtr.Set(KNullDesC);
sl@0
   363
sl@0
   364
            TInt length = iReverseFormattedNumber->Length();
sl@0
   365
            if( aTo < length + 1 )
sl@0
   366
                {
sl@0
   367
                // Advance to the next non-space
sl@0
   368
                if( iExtension->iNumberGroupingCRValue )
sl@0
   369
                	{
sl@0
   370
					while( (aFrom < length ) && (*iReverseFormattedNumber)[aFrom] == KNumberGroupingSpace )
sl@0
   371
						{
sl@0
   372
						++aFrom;
sl@0
   373
						}
sl@0
   374
	
sl@0
   375
					// Retreat to the last non-space
sl@0
   376
					while( (aTo > 0) && (*iReverseFormattedNumber)[aTo] == KNumberGroupingSpace )
sl@0
   377
						{
sl@0
   378
						--aTo;
sl@0
   379
						}
sl@0
   380
                	}
sl@0
   381
sl@0
   382
                // Does fetching the descriptor still make sense?
sl@0
   383
                if ( (0 <= aFrom) && (aFrom <= aTo) && (aTo < length) )
sl@0
   384
                    iReverseFormattedNumberPtr.Set(
sl@0
   385
                        iReverseFormattedNumber->Mid( aFrom, aTo-aFrom+1) );
sl@0
   386
                }
sl@0
   387
            }
sl@0
   388
        else
sl@0
   389
            iReverseFormattedNumberPtr.Set(KNullDesC);
sl@0
   390
        }
sl@0
   391
sl@0
   392
    return iReverseFormattedNumberPtr; // Zero initialized at construction
sl@0
   393
    }
sl@0
   394
sl@0
   395
EXPORT_C const TDesC& CPNGNumberGrouping::ReverseFormattedNumber() const
sl@0
   396
    {
sl@0
   397
    if( iReverseFormattedNumber && !iReverseFormattedNumber->Length())
sl@0
   398
        {
sl@0
   399
        if(!iFormattedNumber->Length())
sl@0
   400
            FormattedNumber();
sl@0
   401
sl@0
   402
        TInt nLength = iFormattedNumber->Length();
sl@0
   403
sl@0
   404
        TPtr ptrModifyable(iReverseFormattedNumber->Des());
sl@0
   405
        TBuf<1> bufChar(1);
sl@0
   406
sl@0
   407
        for(TInt i = nLength; i > 0; --i)
sl@0
   408
            {
sl@0
   409
            TText cChar = (*iFormattedNumber)[i-1];
sl@0
   410
            bufChar[0] = cChar;
sl@0
   411
            ptrModifyable.Insert(nLength - i, bufChar);
sl@0
   412
            }
sl@0
   413
sl@0
   414
        iReverseFormattedNumberPtr.Set(iReverseFormattedNumber->Ptr(), nLength);
sl@0
   415
        }
sl@0
   416
sl@0
   417
    return iReverseFormattedNumberPtr;
sl@0
   418
    }
sl@0
   419
sl@0
   420
EXPORT_C const TDesC& CPNGNumberGrouping::Selection(TInt aFrom, TInt aTo) const
sl@0
   421
    {
sl@0
   422
    if(aFrom < iUnformattedNumber->Length())
sl@0
   423
        {
sl@0
   424
        TPtr ptrUnformatted = iUnformattedNumber->Des();
sl@0
   425
        iSelectionPtr.Set(&(ptrUnformatted[aFrom]), aTo - aFrom);
sl@0
   426
        }
sl@0
   427
    else
sl@0
   428
        iSelectionPtr.Set(KNullDesC);
sl@0
   429
sl@0
   430
    return iSelectionPtr;
sl@0
   431
    }
sl@0
   432
sl@0
   433
EXPORT_C const TDesC&   CPNGNumberGrouping::UnFormattedNumber(TInt aFrom, TInt aTo) const
sl@0
   434
    {
sl@0
   435
    if (iUnformattedNumber && aFrom >= 0 && aFrom <= aTo && aTo < iUnformattedNumber->Length())
sl@0
   436
        {
sl@0
   437
        iUnformattedNumberPtr.Set(&((*iUnformattedNumber)[aFrom]), aTo - aFrom + 1);
sl@0
   438
        }
sl@0
   439
    else
sl@0
   440
        {
sl@0
   441
        iUnformattedNumberPtr.Set(KNullDesC);
sl@0
   442
        }
sl@0
   443
    return iUnformattedNumberPtr;
sl@0
   444
    }
sl@0
   445
sl@0
   446
EXPORT_C const TDesC& CPNGNumberGrouping::UnFormattedNumber() const
sl@0
   447
    {
sl@0
   448
    return UnFormattedNumber(0, iUnformattedNumber->Length() - 1);
sl@0
   449
    }
sl@0
   450
sl@0
   451
TLanguage CPNGNumberGrouping::doReadLanguageFromSharedData() const
sl@0
   452
    {    
sl@0
   453
    if (iExtension->iNumberGroupingCRValue)
sl@0
   454
        {
sl@0
   455
        return ELangAmerican;
sl@0
   456
        }
sl@0
   457
    else
sl@0
   458
        {
sl@0
   459
        return ELangTest;
sl@0
   460
        }
sl@0
   461
    }
sl@0
   462
sl@0
   463
void CPNGNumberGrouping::doClearFormattedNumbers()
sl@0
   464
    {
sl@0
   465
    TPtr ptrModifyable( iUnformattedNumber->Des() );
sl@0
   466
sl@0
   467
    for (TInt index = 0; index < ptrModifyable.Length(); index++)
sl@0
   468
        {
sl@0
   469
        TChar ch = TChar(ptrModifyable[index]);
sl@0
   470
        ch.Fold( TChar::EFoldDigits | TChar::EFoldSpaces);
sl@0
   471
        }
sl@0
   472
    
sl@0
   473
    iFormattedNumber->Des().Zero();
sl@0
   474
    iFormattedNumberPtr.Set(KNullDesC);
sl@0
   475
sl@0
   476
    if ( iReverseFormattedNumber )
sl@0
   477
        iReverseFormattedNumber->Des().Zero();
sl@0
   478
sl@0
   479
    iReverseFormattedNumberPtr.Set(KNullDesC);
sl@0
   480
    iMatchedPatternIndex = ENoMatchedPattern;
sl@0
   481
    }
sl@0
   482
sl@0
   483
void CPNGNumberGrouping::doReadFormatInfoFromResourceFileL()
sl@0
   484
    {
sl@0
   485
    doClearGroupingItemsList();
sl@0
   486
    delete iRegExp;
sl@0
   487
    iRegExp = NULL;
sl@0
   488
sl@0
   489
    RPointerArray<TDesC> parrGroupingPatternsList;
sl@0
   490
    CleanupResetAndDestroyPushL(parrGroupingPatternsList);
sl@0
   491
sl@0
   492
    TInt maxExtraCharacters(0);
sl@0
   493
sl@0
   494
    RFs fs;
sl@0
   495
    CleanupClosePushL(fs);
sl@0
   496
    if(fs.Connect() == KErrNone)
sl@0
   497
        {
sl@0
   498
        RResourceFile resourceFile;
sl@0
   499
        CleanupClosePushL(resourceFile);
sl@0
   500
sl@0
   501
        resourceFile.OpenL(fs, _L("z:\\resource\\numbergrouping.rsc"));
sl@0
   502
        HBufC8* bufResource = resourceFile.AllocReadL(R_GROUPING_MAPPING);
sl@0
   503
sl@0
   504
        TResourceReader resourceReader;
sl@0
   505
        resourceReader.SetBuffer(bufResource);
sl@0
   506
sl@0
   507
        TInt    nLanguageCount = resourceReader.ReadInt8();
sl@0
   508
        TBool   bLanguageMatches = EFalse;
sl@0
   509
sl@0
   510
        while(nLanguageCount-- || !bLanguageMatches)
sl@0
   511
            {
sl@0
   512
            TBool bLanguageMatches = (resourceReader.ReadInt8() == iLanguage);
sl@0
   513
sl@0
   514
            if(bLanguageMatches || ((nLanguageCount == -1) && !bLanguageMatches))
sl@0
   515
                {
sl@0
   516
                TInt nGroupingSchemeCount = resourceReader.ReadInt8();
sl@0
   517
sl@0
   518
                while(nGroupingSchemeCount--)
sl@0
   519
                    {
sl@0
   520
                    TInt thisMaxExtraCharacters(0);
sl@0
   521
                    ReadGroupingSchemeL(
sl@0
   522
                        resourceReader, parrGroupingPatternsList, thisMaxExtraCharacters );
sl@0
   523
                    // take this new max extra characters if bigger
sl@0
   524
                    maxExtraCharacters = Max( maxExtraCharacters, thisMaxExtraCharacters );
sl@0
   525
                    }
sl@0
   526
sl@0
   527
                break; // This breaks out because we take the first language that matches
sl@0
   528
sl@0
   529
                } // End of if on language/locale test
sl@0
   530
            else  // skip other locales
sl@0
   531
                {
sl@0
   532
                TInt nGroupingSchemeCount = resourceReader.ReadInt8();
sl@0
   533
                while(nGroupingSchemeCount--)
sl@0
   534
                    {
sl@0
   535
                    SkipGroupingSchemeL( resourceReader );
sl@0
   536
                    }
sl@0
   537
                }
sl@0
   538
            }
sl@0
   539
sl@0
   540
        delete bufResource;
sl@0
   541
sl@0
   542
        resourceFile.Close();
sl@0
   543
        CleanupStack::Pop();  // resource file
sl@0
   544
        }
sl@0
   545
sl@0
   546
    fs.Close();
sl@0
   547
    CleanupStack::Pop();  // file system
sl@0
   548
sl@0
   549
    iExtension->iMaxExtraCharacters = maxExtraCharacters; // Latch the high water mark of extra characters
sl@0
   550
sl@0
   551
    iRegExp = CRegularExpression::NewL(&parrGroupingPatternsList);
sl@0
   552
sl@0
   553
    CleanupStack::PopAndDestroy(&parrGroupingPatternsList);  // patterns list
sl@0
   554
    }
sl@0
   555
sl@0
   556
void CPNGNumberGrouping::doNumberGroupingL() const
sl@0
   557
    {
sl@0
   558
    TInt lengthToGroup = LengthToGroup();
sl@0
   559
sl@0
   560
    if ( lengthToGroup >= KMinimumLengthToGroup )
sl@0
   561
        {
sl@0
   562
sl@0
   563
        TInt matchedPattern = KErrNotFound;
sl@0
   564
        TInt newMatchedPattern = KErrNotFound;
sl@0
   565
sl@0
   566
        // Search for matches in the RegExp object. It returns the next matching pattern
sl@0
   567
        // However, even if there is a match, lengthToGroup may not be in the deployment
sl@0
   568
        // length range between minDigits and MaxDigits, inclusive
sl@0
   569
        do  {
sl@0
   570
            // Check for another matching pattern
sl@0
   571
            newMatchedPattern = iRegExp->SearchFrom( newMatchedPattern+1, *iUnformattedNumber);
sl@0
   572
sl@0
   573
            if( newMatchedPattern != KErrNotFound) // Found a match, but it is OK?
sl@0
   574
                {
sl@0
   575
sl@0
   576
                TInt minDigits = iGroupingItemsList[newMatchedPattern]->iMinNumberOfDigits;
sl@0
   577
                TInt maxDigits = iGroupingItemsList[newMatchedPattern]->iMaxNumberOfDigits;
sl@0
   578
sl@0
   579
                // Fill in sensible values for min and max if not present
sl@0
   580
                if(minDigits == -1)
sl@0
   581
                    minDigits = 0;
sl@0
   582
                if(maxDigits == -1)
sl@0
   583
                    maxDigits = lengthToGroup;
sl@0
   584
sl@0
   585
                if ( minDigits <= lengthToGroup && lengthToGroup <= maxDigits )
sl@0
   586
                    {
sl@0
   587
                    matchedPattern = newMatchedPattern; // accept this new pattern
sl@0
   588
                    break;
sl@0
   589
                    }
sl@0
   590
                }
sl@0
   591
sl@0
   592
            } while ( newMatchedPattern != KErrNotFound  );
sl@0
   593
sl@0
   594
        // Actually go and do the grouping
sl@0
   595
        if ( matchedPattern != KErrNotFound )
sl@0
   596
            {
sl@0
   597
            doNumberGroupingForPatternL( matchedPattern, lengthToGroup );
sl@0
   598
            return;
sl@0
   599
            }
sl@0
   600
sl@0
   601
        }
sl@0
   602
sl@0
   603
    // if we get to here, either the string was not matched to any of the patterns or the
sl@0
   604
    // unformatted string is exactly the display length.  In either case we call
sl@0
   605
    // doNumberSquashing() which simply leaves the string as it is...
sl@0
   606
    doNumberSquashing();
sl@0
   607
sl@0
   608
    }
sl@0
   609
sl@0
   610
sl@0
   611
void CPNGNumberGrouping::doNumberGroupingForPatternL( TInt aMatchingPattern, TInt aLengthToGroup ) const
sl@0
   612
    {
sl@0
   613
    iMatchedPatternIndex = aMatchingPattern;
sl@0
   614
sl@0
   615
    TInt nLowPos = 0;
sl@0
   616
    TInt nHighPos = 0;
sl@0
   617
sl@0
   618
    TPtr desUnformattedNumber = iUnformattedNumber->Des();
sl@0
   619
    TInt unformattedLength = iUnformattedNumber->Length();
sl@0
   620
sl@0
   621
    __ASSERT_ALWAYS( aLengthToGroup <= unformattedLength , Panic(ENumberGroupingBadLengthToGroup) );
sl@0
   622
sl@0
   623
    TPNGGroupingInfo* matchedPattern = iGroupingItemsList[iMatchedPatternIndex];
sl@0
   624
    TInt nAfterCount = matchedPattern->iAfterPositions.Count();
sl@0
   625
    TBool bBeforePosition = (matchedPattern->iBeforePosition.iPosition == -1)?0:1;
sl@0
   626
sl@0
   627
    // Test to see if the beforePosition can be used with the current text length.
sl@0
   628
    // The following does not allow the before position to be used if it would result in an
sl@0
   629
    // insertion right next to one from the AfterPositions.
sl@0
   630
    // That is, tildas in the formatting string represent 1 or more characters.
sl@0
   631
    // e.g. if the last afterPosition is 4 and the before position is 3, then a 7 digit
sl@0
   632
    // number will not be able to have the before position used.
sl@0
   633
    if( nAfterCount &&
sl@0
   634
        (unformattedLength - matchedPattern->iBeforePosition.iPosition) <=
sl@0
   635
        matchedPattern->iAfterPositions[nAfterCount - 1].iPosition)
sl@0
   636
        {
sl@0
   637
        bBeforePosition = EFalse;
sl@0
   638
        }
sl@0
   639
sl@0
   640
    TPtr ptrModifyable(iFormattedNumber->Des());
sl@0
   641
sl@0
   642
    for(TInt i  = 0; i < nAfterCount && nHighPos < aLengthToGroup ; ++i)
sl@0
   643
        {
sl@0
   644
        nHighPos = matchedPattern->iAfterPositions[i].iPosition;
sl@0
   645
        if ( nHighPos >= aLengthToGroup )
sl@0
   646
            break;
sl@0
   647
sl@0
   648
        if(nHighPos < unformattedLength)
sl@0
   649
            {
sl@0
   650
            ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
sl@0
   651
            ptrModifyable.Append(matchedPattern->iAfterPositions[i].iSeparatorCharacter);
sl@0
   652
            nLowPos = nHighPos;
sl@0
   653
            }
sl@0
   654
        }
sl@0
   655
sl@0
   656
    // Do not do "before end" formatting at all if there is any truncation
sl@0
   657
    if ( aLengthToGroup < unformattedLength )
sl@0
   658
        {
sl@0
   659
        TInt nBeforePosition = matchedPattern->iBeforePosition.iPosition;
sl@0
   660
sl@0
   661
        if(bBeforePosition && nBeforePosition < unformattedLength)
sl@0
   662
            {
sl@0
   663
            nHighPos = unformattedLength - nBeforePosition;
sl@0
   664
            ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
sl@0
   665
            ptrModifyable.Append( matchedPattern->iBeforePosition.iSeparatorCharacter );
sl@0
   666
            nLowPos = nHighPos;
sl@0
   667
            }
sl@0
   668
        }
sl@0
   669
sl@0
   670
    nHighPos = unformattedLength;
sl@0
   671
    ptrModifyable.Append( desUnformattedNumber.Mid( nLowPos, nHighPos - nLowPos) );
sl@0
   672
sl@0
   673
    }
sl@0
   674
sl@0
   675
void CPNGNumberGrouping::doNumberSquashing() const
sl@0
   676
    {
sl@0
   677
    __ASSERT_ALWAYS( !iFormattedNumber->Length(), Panic(ENumberGroupingFormattedNumberAlreadyExists) );
sl@0
   678
sl@0
   679
    // just copy from one t'other...
sl@0
   680
    TPtr ptrModifyable(iFormattedNumber->Des());
sl@0
   681
    ptrModifyable.Copy(*iUnformattedNumber);
sl@0
   682
    iMatchedPatternIndex = ENoMatchedPattern;
sl@0
   683
    }
sl@0
   684
sl@0
   685
void CPNGNumberGrouping::doClearGroupingItemsList()
sl@0
   686
    {
sl@0
   687
    TInt nCount = iGroupingItemsList.Count();
sl@0
   688
sl@0
   689
    for(TInt i = 0; i < nCount; ++i)
sl@0
   690
        {
sl@0
   691
        iGroupingItemsList[i]->iAfterPositions.Close();
sl@0
   692
        delete iGroupingItemsList[i];
sl@0
   693
        iGroupingItemsList[i] = NULL;
sl@0
   694
        }
sl@0
   695
    iGroupingItemsList.Close();
sl@0
   696
    }
sl@0
   697
sl@0
   698
void CPNGNumberGrouping::ReadGroupingSchemeL(
sl@0
   699
    TResourceReader& aResourceReader,
sl@0
   700
    RPointerArray<TDesC>& aGroupingPatternsList,
sl@0
   701
    TInt& aMaxExtraCharacters )
sl@0
   702
    {
sl@0
   703
    CleanupResetAndDestroyPushL(aGroupingPatternsList);
sl@0
   704
    TPNGGroupingInfo* groupingInfo = new (ELeave) TPNGGroupingInfo;
sl@0
   705
    CleanupStack::PushL( groupingInfo );
sl@0
   706
sl@0
   707
    // Read in all resource for this grouping scheme, perform checking and then analyze it
sl@0
   708
    HBufC* initialDigits = aResourceReader.ReadHBufCL();
sl@0
   709
    __ASSERT_ALWAYS( initialDigits, Panic( ENumberGroupingNoInitialDigitsInResource ) );
sl@0
   710
    CleanupStack::PushL( initialDigits );
sl@0
   711
sl@0
   712
    groupingInfo->iMinNumberOfDigits = aResourceReader.ReadInt8();
sl@0
   713
    groupingInfo->iMaxNumberOfDigits = aResourceReader.ReadInt8();
sl@0
   714
    __ASSERT_DEBUG(
sl@0
   715
        ( groupingInfo->iMaxNumberOfDigits == -1) ||
sl@0
   716
        ( groupingInfo->iMinNumberOfDigits <= groupingInfo->iMaxNumberOfDigits ),
sl@0
   717
        Panic( ENumberGroupingBadMinMaxDigitRangeInResource ) );
sl@0
   718
sl@0
   719
    // Read in formatting Pattern
sl@0
   720
    HBufC* formatPattern = aResourceReader.ReadHBufCL();
sl@0
   721
sl@0
   722
    if ( formatPattern ) // Does not have to be there
sl@0
   723
        {
sl@0
   724
        CleanupStack::PushL( formatPattern );
sl@0
   725
        TInt formatLength = formatPattern->Length();
sl@0
   726
        if ( formatLength > 0 )
sl@0
   727
            {
sl@0
   728
            // Obtain a wildcard version of the matching pattern in initialDigits.
sl@0
   729
            // This is used to check the supplied formatPattern for comformance to initialDigits
sl@0
   730
            HBufC* wildcardedMatchBuf = HBufC::NewLC( formatLength ); // Will not be longer than the search pattern
sl@0
   731
sl@0
   732
            TPtr wildcardedMatchPtr( wildcardedMatchBuf->Des() );
sl@0
   733
            // Get the example number using the latest search pattern only
sl@0
   734
            GetWildcardVersionOfMatchStringL( *initialDigits, KNumberGroupingWildcard, wildcardedMatchPtr );
sl@0
   735
sl@0
   736
            // Now parse the descriptor
sl@0
   737
            TBool trailingPossible(EFalse);
sl@0
   738
            ParseForAfterPositions(
sl@0
   739
                *formatPattern, groupingInfo, wildcardedMatchPtr, aMaxExtraCharacters, trailingPossible );
sl@0
   740
sl@0
   741
            // Now parse the descriptor from the end if needed
sl@0
   742
            if ( trailingPossible )
sl@0
   743
                ParseForBeforePosition( *formatPattern, groupingInfo, aMaxExtraCharacters );
sl@0
   744
sl@0
   745
            CleanupStack::PopAndDestroy( wildcardedMatchBuf );
sl@0
   746
            }
sl@0
   747
        CleanupStack::PopAndDestroy( formatPattern );
sl@0
   748
        } // End of if on formatPattern.Length
sl@0
   749
sl@0
   750
    User::LeaveIfError( aGroupingPatternsList.Append( initialDigits ) );
sl@0
   751
    CleanupStack::Pop( initialDigits );
sl@0
   752
sl@0
   753
    // Do not leave if the next one fails, but remove the last from the patterns list and then leave
sl@0
   754
    // This is done in case someone TRAPs. Otherwise neither of these lists would be used and their
sl@0
   755
    // mismatch would not be a problem
sl@0
   756
    if ( TInt err = iGroupingItemsList.Append(groupingInfo) != KErrNone )
sl@0
   757
        {
sl@0
   758
        // return value of Count will be at least 1, because we have just successfully gone through an Append
sl@0
   759
        aGroupingPatternsList.Remove( aGroupingPatternsList.Count() - 1 );
sl@0
   760
        // ownership is now mine again...
sl@0
   761
        delete initialDigits;
sl@0
   762
        // Need to delete groupingInfo, and make sure it is no longer on the cleanupstack
sl@0
   763
        CleanupStack::PopAndDestroy( groupingInfo );
sl@0
   764
        User::Leave(err);
sl@0
   765
        }
sl@0
   766
    else
sl@0
   767
        {
sl@0
   768
        CleanupStack::Pop( groupingInfo ); // Success. This object now not owned by the cleanupstack
sl@0
   769
        }
sl@0
   770
    
sl@0
   771
    CleanupStack::Pop(&aGroupingPatternsList);
sl@0
   772
    }
sl@0
   773
sl@0
   774
void CPNGNumberGrouping::ParseForAfterPositions(
sl@0
   775
    const TDesC& aFormatPattern,
sl@0
   776
    TPNGGroupingInfo* aGroupingInfo,
sl@0
   777
    const TDesC& aWildcardedMatchingPattern,
sl@0
   778
    TInt& aMaxExtraCharacters,
sl@0
   779
    TBool& trailingPossible ) const
sl@0
   780
    {
sl@0
   781
    TInt pos(0); // Keeps track of the position with which the next separator will be stored
sl@0
   782
    TInt formatLength = aFormatPattern.Length();
sl@0
   783
    for (TInt index = 0; index < formatLength; index++ )
sl@0
   784
        {
sl@0
   785
        // The format pattern is compared with the matching pattern.  The matching pattern may be
sl@0
   786
        // shorter than the format pattern, so by default a wildcard character is used.
sl@0
   787
        TText ch = aFormatPattern[index];
sl@0
   788
        TText matchingChar(KNumberGroupingWildcard); // default to expect is the wildcard character
sl@0
   789
        if ( pos < aWildcardedMatchingPattern.Length() ) // if still within the matching pattern
sl@0
   790
            matchingChar = aWildcardedMatchingPattern[pos];
sl@0
   791
        if ( ch == matchingChar )
sl@0
   792
            pos++; // not a separator. index where the next "after" marker goes
sl@0
   793
        else if ( ch == KNumberGroupingOneOrMoreCharactersToken )
sl@0
   794
            {
sl@0
   795
            // finish looking for "afterPositions". But there may be a "before" position in the
sl@0
   796
            // remainder, so set the flag
sl@0
   797
            trailingPossible = ETrue;
sl@0
   798
            break;
sl@0
   799
            }
sl@0
   800
        else
sl@0
   801
            {
sl@0
   802
            // Explicit prevention of any separator characters being valid phone numbers
sl@0
   803
#ifdef _DEBUG
sl@0
   804
            if ( IsValidPhoneNumberCharacter( ch ) || ch == KNumberGroupingWildcard )
sl@0
   805
                {
sl@0
   806
                RDebug::Print(
sl@0
   807
                    _L("NumberGrouping: Illegal character or format mismatch in resource: initialDigits pattern= <%S> formatPattern=<%S>"),
sl@0
   808
                    &aWildcardedMatchingPattern, &aFormatPattern );
sl@0
   809
                }
sl@0
   810
#endif
sl@0
   811
            __ASSERT_DEBUG( !IsValidPhoneNumberCharacter( ch ), Panic( ENumberGroupingInvalidSeparatorCharacterInFormat ) );
sl@0
   812
            __ASSERT_DEBUG( ch != KNumberGroupingWildcard, Panic( ENumberGroupingMatchingPatternVersusFormatPatternMismatch ) );
sl@0
   813
            TPNGSeparator separator( pos, aFormatPattern[index]);
sl@0
   814
            aGroupingInfo->iAfterPositions.Append(separator);
sl@0
   815
            aMaxExtraCharacters++;
sl@0
   816
            }
sl@0
   817
        }
sl@0
   818
    }
sl@0
   819
sl@0
   820
void CPNGNumberGrouping::ParseForBeforePosition(
sl@0
   821
    const TDesC& aFormatPattern,
sl@0
   822
    TPNGGroupingInfo* aGroupingInfo,
sl@0
   823
    TInt& aMaxExtraCharacters ) const
sl@0
   824
    {
sl@0
   825
    TInt pos=0;
sl@0
   826
    TInt formatLength = aFormatPattern.Length();
sl@0
   827
sl@0
   828
    for (TInt index = formatLength-1; index >=0; index-- )
sl@0
   829
        {
sl@0
   830
        TText ch = aFormatPattern[index];
sl@0
   831
        if ( ch == KNumberGroupingWildcard )
sl@0
   832
            pos++;
sl@0
   833
        else if ( ch == KNumberGroupingOneOrMoreCharactersToken )
sl@0
   834
            break;
sl@0
   835
        else
sl@0
   836
            {
sl@0
   837
            // Explicit prevention of any separator characters being valid phone numbers
sl@0
   838
#ifdef _DEBUG
sl@0
   839
            if ( IsValidPhoneNumberCharacter( ch ) )
sl@0
   840
                {
sl@0
   841
                RDebug::Print(
sl@0
   842
                    _L("NumberGrouping: Illegal character in trailing part of format string in resource: formatPattern=<%S>"),
sl@0
   843
                    &aFormatPattern );
sl@0
   844
                }
sl@0
   845
#endif
sl@0
   846
            __ASSERT_DEBUG( !IsValidPhoneNumberCharacter( ch ),
sl@0
   847
                Panic( ENumberGroupingInvalidSeparatorCharacterInFormat ) );
sl@0
   848
            TPNGSeparator separator( pos, ch );
sl@0
   849
            aGroupingInfo->iBeforePosition = separator;
sl@0
   850
            aMaxExtraCharacters++;
sl@0
   851
            break;
sl@0
   852
            }
sl@0
   853
        }
sl@0
   854
    }
sl@0
   855
sl@0
   856
sl@0
   857
void CPNGNumberGrouping::SkipGroupingSchemeL( TResourceReader& aResourceReader ) const
sl@0
   858
    {
sl@0
   859
    HBufC* tempBuf;
sl@0
   860
    tempBuf = aResourceReader.ReadHBufCL();
sl@0
   861
    delete tempBuf;
sl@0
   862
    aResourceReader.Advance(2); // min and max characters
sl@0
   863
    tempBuf = aResourceReader.ReadHBufCL();
sl@0
   864
    delete tempBuf;
sl@0
   865
    }
sl@0
   866
sl@0
   867
void CPNGNumberGrouping::GetWildcardVersionOfMatchStringL(
sl@0
   868
    const TDesC& aMatchString,
sl@0
   869
    TText aWildcard,
sl@0
   870
    TDes& aWildcardMatchString ) const
sl@0
   871
    {
sl@0
   872
    RPointerArray<TDesC> patternList;
sl@0
   873
    CleanupClosePushL(patternList);
sl@0
   874
sl@0
   875
    // Make a copy of the input string
sl@0
   876
    HBufC* matchString = aMatchString.AllocLC();
sl@0
   877
sl@0
   878
    User::LeaveIfError( patternList.Append(matchString) );// takes ownership
sl@0
   879
    CleanupStack::Pop( matchString );
sl@0
   880
sl@0
   881
    CRegularExpression* regExp = CRegularExpression::NewLC(&patternList);
sl@0
   882
sl@0
   883
    // Only 1 pattern fed in.  Access that pattern at index 0
sl@0
   884
    regExp->GetWildcardVersionOfPattern( 0 , aWildcard, aWildcardMatchString );
sl@0
   885
sl@0
   886
    CleanupStack::PopAndDestroy(regExp);
sl@0
   887
sl@0
   888
    // Delete the patterns list
sl@0
   889
    delete patternList[0];
sl@0
   890
    CleanupStack::PopAndDestroy();
sl@0
   891
    }
sl@0
   892
sl@0
   893
sl@0
   894
EXPORT_C TBool CPNGNumberGrouping::IsCharacterInsertedByNumberGrouping(TInt aPos) const
sl@0
   895
    {
sl@0
   896
    TInt insertedCharacters = Length() - UnFormattedLength();
sl@0
   897
sl@0
   898
    if( insertedCharacters == 0 ) // no formatting was done
sl@0
   899
        return EFalse;
sl@0
   900
    else if ( aPos < ( insertedCharacters + LengthToGroup() ) )
sl@0
   901
        {
sl@0
   902
        return !IsValidPhoneNumberCharacter( (*iFormattedNumber)[aPos] );
sl@0
   903
        }
sl@0
   904
    else // aPos is pointing at or beyond index= LengthToGroup() + <chars inserted>; no formatting there
sl@0
   905
        return EFalse;
sl@0
   906
    }
sl@0
   907
sl@0
   908
sl@0
   909
TBool CPNGNumberGrouping::IsValidPhoneNumberCharacter( TText aCharacter ) const
sl@0
   910
    {
sl@0
   911
    if ( ((TChar)aCharacter).IsDigit() )
sl@0
   912
        return ETrue;
sl@0
   913
sl@0
   914
    // Check through the list of additional valid phone number characters
sl@0
   915
    TInt numAdditionalChars = sizeof( KAdditionalPhoneNumberCharacters )/sizeof(TText);
sl@0
   916
sl@0
   917
    for (TInt index = 0; index < numAdditionalChars; index++)
sl@0
   918
        {
sl@0
   919
        if ( aCharacter == KAdditionalPhoneNumberCharacters[index] )
sl@0
   920
            return ETrue;
sl@0
   921
        }
sl@0
   922
sl@0
   923
    return EFalse;
sl@0
   924
    }
sl@0
   925
sl@0
   926
EXPORT_C TBool CPNGNumberGrouping::IsChangedByGrouping() const
sl@0
   927
    {
sl@0
   928
    // The only way that grouping is effectively different is by making things longer
sl@0
   929
    return ( Length() > UnFormattedLength() );
sl@0
   930
    }
sl@0
   931
sl@0
   932
TInt CPNGNumberGrouping::LengthToGroup() const
sl@0
   933
    {
sl@0
   934
sl@0
   935
    TPtrC ptr = iUnformattedNumber->Des();
sl@0
   936
    TInt lengthToGroup = ptr.Length();
sl@0
   937
sl@0
   938
    // Find the first non-digit
sl@0
   939
    for (TInt index = 0; index < ptr.Length(); index++)
sl@0
   940
        {
sl@0
   941
        TChar ch = TChar(ptr[index]);
sl@0
   942
        ch.Fold(TChar::EFoldDigits);
sl@0
   943
        if ( !( ch.IsDigit() ) )
sl@0
   944
            {
sl@0
   945
            lengthToGroup = index; // only characters BEFORE the character at index are grouped
sl@0
   946
            break;
sl@0
   947
            }
sl@0
   948
        }
sl@0
   949
sl@0
   950
    return lengthToGroup;
sl@0
   951
    }
sl@0
   952
sl@0
   953
// End of File