os/ossrv/genericservices/s60compatibilityheaders/commonengine/src/textresolver.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
sl@0
    17
sl@0
    18
#include <barsread.h>    // TResourceReader
sl@0
    19
#include <bautils.h>     // BaflUtils
sl@0
    20
#include "textresolver.h"
sl@0
    21
#include <errorres.rsg>
sl@0
    22
#include <uikon/eikdefconst.h>
sl@0
    23
sl@0
    24
sl@0
    25
const TInt KUnknownError( 0 );
sl@0
    26
const TInt KBlankError( -2 );
sl@0
    27
sl@0
    28
const TInt KRIMask(0x00000fff);
sl@0
    29
_LIT(KErrorResResourceFileName, "z:\\resource\\errors\\ERRORRES.RSC"); 
sl@0
    30
_LIT(KRDSupport, "c:\\resource\\ErrRD");
sl@0
    31
_LIT(KTextResolverPanic,   "TextResolver");
sl@0
    32
sl@0
    33
sl@0
    34
GLDEF_C void Panic(TInt aPanic)
sl@0
    35
	{
sl@0
    36
    User::Panic(KTextResolverPanic, aPanic);
sl@0
    37
	}
sl@0
    38
sl@0
    39
sl@0
    40
CTextResolver::CTextResolver()
sl@0
    41
    {
sl@0
    42
    }
sl@0
    43
sl@0
    44
CTextResolver::CTextResolver(CCoeEnv& aEnv)
sl@0
    45
    :iCoe(&aEnv)
sl@0
    46
    {
sl@0
    47
    }
sl@0
    48
/**
sl@0
    49
 * Two-phase constructor method that is used to create a new instance
sl@0
    50
 * of the CTextResolver class. The implementation uses the passed 
sl@0
    51
 * CCoeEnv instance to access the resource files.
sl@0
    52
 *
sl@0
    53
 * @param aEnv the CCoeEnv instance to be used to access the resource
sl@0
    54
 * files.
sl@0
    55
 * @return a pointer to a new instance of the CTextResolver class.
sl@0
    56
 */
sl@0
    57
EXPORT_C CTextResolver* CTextResolver::NewL(CCoeEnv& aEnv)
sl@0
    58
    {
sl@0
    59
    CTextResolver* self = CTextResolver::NewLC(aEnv);
sl@0
    60
    CleanupStack::Pop(self);
sl@0
    61
    return self;
sl@0
    62
    }
sl@0
    63
sl@0
    64
 /**
sl@0
    65
 * Constructor creates a new instance of CTextResolver. The 
sl@0
    66
 * implementation uses the passed CCoeEnv instance to access the 
sl@0
    67
 * resource files. Leaves the object on the cleanup stack.
sl@0
    68
 *
sl@0
    69
 * @param aEnv the CCoeEnv instance to be used to access the resource
sl@0
    70
 * files.
sl@0
    71
 * @return a pointer to a new instance of the CTextResolver class.
sl@0
    72
 */
sl@0
    73
EXPORT_C CTextResolver* CTextResolver::NewLC(CCoeEnv& aEnv)
sl@0
    74
    {
sl@0
    75
    CTextResolver* self = new (ELeave) CTextResolver(aEnv);
sl@0
    76
    CleanupStack::PushL(self);
sl@0
    77
    self->ConstructL();
sl@0
    78
    return self;
sl@0
    79
    }
sl@0
    80
sl@0
    81
 /**
sl@0
    82
 * Constructor creates a new instance of CTextResolver. Resource files 
sl@0
    83
 * are accessed through the RResourceFile API.
sl@0
    84
 *
sl@0
    85
 * @return a pointer to a new instance of the CTextResolver class.
sl@0
    86
 */
sl@0
    87
EXPORT_C CTextResolver* CTextResolver::NewL()
sl@0
    88
    {
sl@0
    89
    CTextResolver* self = CTextResolver::NewLC();
sl@0
    90
    CleanupStack::Pop(self);
sl@0
    91
    return self;
sl@0
    92
    }
sl@0
    93
sl@0
    94
 /**
sl@0
    95
 * Constructor creates a new instance of CTextResolver.Resource files 
sl@0
    96
 * are accessed through the RResourceFile API. Leaves the object on 
sl@0
    97
 * the cleanup stack.
sl@0
    98
 *
sl@0
    99
 * @return a pointer to a new instance of the CTextResolver class.
sl@0
   100
 */
sl@0
   101
EXPORT_C CTextResolver* CTextResolver::NewLC()
sl@0
   102
    {
sl@0
   103
    CTextResolver* self = new (ELeave) CTextResolver();
sl@0
   104
    CleanupStack::PushL(self);
sl@0
   105
    self->ConstructL();
sl@0
   106
    return self;
sl@0
   107
    }
sl@0
   108
sl@0
   109
//
sl@0
   110
// ---------------------------------------------------------
sl@0
   111
// CTextResolver::ConstructL
sl@0
   112
//
sl@0
   113
// Symbian OS default constructor, initializes variables and cache 
sl@0
   114
// ---------------------------------------------------------
sl@0
   115
//
sl@0
   116
void CTextResolver::ConstructL()
sl@0
   117
    {
sl@0
   118
    // if no coe is present, connect new filesession.
sl@0
   119
    if (!iCoe)
sl@0
   120
    	{
sl@0
   121
        User::LeaveIfError(iFs.Connect());    
sl@0
   122
sl@0
   123
		TFileName resFile(KErrorResResourceFileName);
sl@0
   124
		BaflUtils::NearestLanguageFile(iFs, resFile);
sl@0
   125
		iResFile.OpenL(iFs, resFile);
sl@0
   126
    	}
sl@0
   127
sl@0
   128
    iRDSupport = BaflUtils::FileExists(iCoe ? iCoe->FsSession() : iFs, KRDSupport);     
sl@0
   129
    }
sl@0
   130
sl@0
   131
/** 
sl@0
   132
* Destructor 
sl@0
   133
*/
sl@0
   134
EXPORT_C CTextResolver::~CTextResolver()
sl@0
   135
    {
sl@0
   136
    Reset();
sl@0
   137
    iResFile.Close();
sl@0
   138
sl@0
   139
    // if no coe is present, close filesession.
sl@0
   140
    if (!iCoe)
sl@0
   141
        iFs.Close();   
sl@0
   142
    
sl@0
   143
    delete iTitleText;
sl@0
   144
    delete iTextBuffer;
sl@0
   145
    delete iContextSeparator;
sl@0
   146
    }
sl@0
   147
sl@0
   148
/**
sl@0
   149
* Resolves the given error code and returns the error text for the
sl@0
   150
* resolved error. Resolved text can be of any length. This version 
sl@0
   151
* is for "normal" use.
sl@0
   152
*
sl@0
   153
* @param aError The error code to be resolved.
sl@0
   154
* @param aContext Optional context for error numbers. If the aContext 
sl@0
   155
* parameter is not passed to the function, it uses the default value
sl@0
   156
* ECtxAutomatic. 
sl@0
   157
* @return the error text for the resolved error. "System error" (in 
sl@0
   158
* English localisation) is returned when error code is not known. For 
sl@0
   159
* unknown errors blank error flag (flags are defined in 
sl@0
   160
* textresolver.hrh) is also set to hide errors without proper 
sl@0
   161
* explanation. There is no limit on how long the resolved string
sl@0
   162
* can be.
sl@0
   163
*/
sl@0
   164
EXPORT_C const TDesC& CTextResolver::ResolveErrorString(TInt aError, TErrorContext aContext)
sl@0
   165
    {
sl@0
   166
    TUint flags(0);
sl@0
   167
    TInt id(0);
sl@0
   168
	return ResolveErrorString(aError,id,flags,aContext);
sl@0
   169
	}
sl@0
   170
sl@0
   171
/**
sl@0
   172
* Resolves the given error code and returns the error text for the
sl@0
   173
* resolved error. Resolved text can be of any length. This version
sl@0
   174
* is for advanced use
sl@0
   175
*
sl@0
   176
* @param aError The error code to be resolved.
sl@0
   177
* @param aTextId ID of the returned text.
sl@0
   178
* @param aFlags The priority of the returned error. The priority is 
sl@0
   179
* defined by the this module! Flags are defined in textresolver.hrh.
sl@0
   180
* @param aContext Optional context for error numbers. If the aContext 
sl@0
   181
* parameter is not passed to the function, it uses the default value
sl@0
   182
* ECtxAutomatic. 
sl@0
   183
* @return the error text for the resolved error. "System error" (in 
sl@0
   184
* English localisation) is returned when error code is not known. For 
sl@0
   185
* unknown errors blank error flag (flags are defined in 
sl@0
   186
* textresolver.hrh) is also set to hide errors without proper 
sl@0
   187
* explanation. There is no limit on how long the resolved string 
sl@0
   188
* can be.
sl@0
   189
*/
sl@0
   190
EXPORT_C const TDesC& CTextResolver::ResolveErrorString(TInt aError, TInt& aTextId, TUint& aFlags, TErrorContext aContext)
sl@0
   191
    {
sl@0
   192
    aFlags = 0;
sl@0
   193
    if (iCoe)	// Use the resource loaded using the CCoeEnv to read the resource file
sl@0
   194
    	{
sl@0
   195
	    TRAPD(err, DoResolveErrorStringL(aError, aTextId, aFlags));
sl@0
   196
		if (err)
sl@0
   197
			{
sl@0
   198
			if ((!AddContextAndSeparator(aContext)))
sl@0
   199
				{
sl@0
   200
				// Even the resource file opening failed, return OOM flag
sl@0
   201
				aFlags |= EErrorResOOMFlag;
sl@0
   202
				return KNullDesC;
sl@0
   203
				}
sl@0
   204
			return *iTextBuffer;
sl@0
   205
			}            
sl@0
   206
sl@0
   207
		if(aFlags&ETextResolverUnknownErrorFlag)	// Error code not supported. No error text found.
sl@0
   208
	        {
sl@0
   209
	        // Error is not recognised 
sl@0
   210
	        aTextId = R_ERROR_RES_GENERAL;	// Read and return an "unknown error" text from the
sl@0
   211
	        								// Nokia-specific version of the errorres.rss file
sl@0
   212
	      	TPtr appTextPtr = iTitleText->Des();
sl@0
   213
	      	TPtr textPtr = iTextBuffer->Des();
sl@0
   214
	      	iCoe->ReadResourceL(appTextPtr, R_BASE);
sl@0
   215
	      	iCoe->ReadResourceL(textPtr, R_ERROR_RES_GENERAL);
sl@0
   216
	        }
sl@0
   217
		}
sl@0
   218
	else	// No CCoeEnv available
sl@0
   219
		{
sl@0
   220
		aTextId = 0;
sl@0
   221
        TRAPD(err, DoRawReadOfSystemErrorResourcesToArraysL(aError, aTextId));
sl@0
   222
        if(err)
sl@0
   223
            {
sl@0
   224
            // Cleaning needed since object not destroyed
sl@0
   225
            Reset();
sl@0
   226
            
sl@0
   227
            if (aTextId != R_ERROR_RES_NO_MEMORY || !AddContextAndSeparator(aContext))
sl@0
   228
                {
sl@0
   229
                // Even the resource file opening failed, return OOM flag
sl@0
   230
                aFlags |= EErrorResOOMFlag;
sl@0
   231
                return KNullDesC;
sl@0
   232
                }
sl@0
   233
sl@0
   234
            return *iTextBuffer;
sl@0
   235
            }            
sl@0
   236
        
sl@0
   237
        aTextId = ResourceForError(aError);
sl@0
   238
        switch (aTextId)
sl@0
   239
            {
sl@0
   240
            case KBlankError:
sl@0
   241
                {
sl@0
   242
                aFlags |= ETextResolverBlankErrorFlag;
sl@0
   243
                break;
sl@0
   244
                }
sl@0
   245
            case KUnknownError:
sl@0
   246
                {
sl@0
   247
                // Error is not recognised 
sl@0
   248
                aTextId = R_ERROR_RES_GENERAL;
sl@0
   249
sl@0
   250
				delete iTitleText;
sl@0
   251
                iTitleText = AllocReadUnicodeString(iResFile, R_BASE);
sl@0
   252
                aFlags |= ETextResolverUnknownErrorFlag;
sl@0
   253
                // No break here; Fall through to default case to resolve text for the general error
sl@0
   254
                }
sl@0
   255
            default:    
sl@0
   256
                {
sl@0
   257
                // --- Error recognised by the resolver, so get the text ---
sl@0
   258
				delete iTextBuffer;
sl@0
   259
                iTextBuffer = AllocReadUnicodeString(iResFile, aTextId);
sl@0
   260
                break;
sl@0
   261
                }
sl@0
   262
            };       
sl@0
   263
        }       
sl@0
   264
sl@0
   265
	if(aFlags&ETextResolverBlankErrorFlag)	// Error code supported, but no error text found.
sl@0
   266
		{
sl@0
   267
		delete iTextBuffer;
sl@0
   268
		iTextBuffer = NULL;				// Return no error text at all	
sl@0
   269
		}
sl@0
   270
sl@0
   271
    if (!AddContextAndSeparator(aContext))
sl@0
   272
		{
sl@0
   273
		aFlags |= EErrorResOOMFlag;
sl@0
   274
		return KNullDesC;
sl@0
   275
		}
sl@0
   276
sl@0
   277
    if (iRDSupport && iTextBuffer) 
sl@0
   278
        {
sl@0
   279
        HBufC* tempBuffer = iTextBuffer->ReAlloc(iTextBuffer->Length()+14);
sl@0
   280
sl@0
   281
        // if allocation succeeds, append error id. Otherwise iTextBuffer stays as it is
sl@0
   282
        if (tempBuffer)
sl@0
   283
            {
sl@0
   284
            iTextBuffer = tempBuffer;
sl@0
   285
            iTextBuffer->Des().Append(' ');
sl@0
   286
            iTextBuffer->Des().Append('(');
sl@0
   287
            iTextBuffer->Des().AppendNum(aError);
sl@0
   288
            iTextBuffer->Des().Append(')');
sl@0
   289
            }
sl@0
   290
        }
sl@0
   291
	else if (aFlags & ETextResolverUnknownErrorFlag)
sl@0
   292
		{ // We hide the errors we cannot give proper explanation, 
sl@0
   293
		  //i.e. "System Error" will not be shown.
sl@0
   294
		aFlags |= ETextResolverBlankErrorFlag;	
sl@0
   295
        }
sl@0
   296
sl@0
   297
    if (iTextBuffer)
sl@0
   298
        return *iTextBuffer;
sl@0
   299
    else
sl@0
   300
        return KNullDesC;
sl@0
   301
    }
sl@0
   302
sl@0
   303
TInt CTextResolver::ResourceForError(TInt aError)
sl@0
   304
    {
sl@0
   305
    ASSERT(!iCoe);
sl@0
   306
    const TInt errorCount = iStartError->Count();
sl@0
   307
            
sl@0
   308
    delete iTitleText;
sl@0
   309
    iTitleText = NULL;
sl@0
   310
    
sl@0
   311
    for (TInt cc = 0; cc < errorCount; ++cc)
sl@0
   312
        {
sl@0
   313
        const TInt starterror = (*iStartError)[cc];
sl@0
   314
        const TInt endError = starterror-(*iErrorTexts)[cc]->Count()+1;
sl@0
   315
        
sl@0
   316
        if (aError >= endError && aError <= starterror)
sl@0
   317
        	{
sl@0
   318
	        ASSERT(!iTitleText);
sl@0
   319
	        iTitleText = AllocReadUnicodeString(iResFile, (*iAppTexts)[cc]);
sl@0
   320
sl@0
   321
	        const TInt errorIndex = -(aError-starterror);
sl@0
   322
	        const TInt flags = (*iFlags)[cc]->At(errorIndex);
sl@0
   323
	        
sl@0
   324
	        if (flags & ETextResolverPanicErrorFlag)
sl@0
   325
	            {
sl@0
   326
	            // --- Some errors are never meant to get to the UI level ---
sl@0
   327
	            // --- Those of them that do, result in a panic ---
sl@0
   328
	            Panic(aError);
sl@0
   329
	            }
sl@0
   330
	        
sl@0
   331
	        else if (flags & ETextResolverBlankErrorFlag)
sl@0
   332
	            {
sl@0
   333
	            // --- Error code can be ignored from the users' perspective ---
sl@0
   334
	            // --- Return blank error text ---
sl@0
   335
	            return KBlankError;
sl@0
   336
	            }
sl@0
   337
sl@0
   338
	        else if (flags & ETextResolverUnknownErrorFlag)
sl@0
   339
	            {
sl@0
   340
	            // --- Error not recognised by TextResolver ---
sl@0
   341
	            // --- Will be passed on to UIKON for interpretation ---
sl@0
   342
	            return KUnknownError;
sl@0
   343
	            }
sl@0
   344
	            
sl@0
   345
	        return (*iErrorTexts)[cc]->At(errorIndex); 
sl@0
   346
        	}
sl@0
   347
        }
sl@0
   348
        
sl@0
   349
   return KUnknownError;     
sl@0
   350
   }
sl@0
   351
    
sl@0
   352
sl@0
   353
/**
sl@0
   354
Read the system error texts from errorres.rss only, without using the Uikon CEikErrorResolver.
sl@0
   355
This will not pick up any appliation specific errors.
sl@0
   356
*/
sl@0
   357
void CTextResolver::DoRawReadOfSystemErrorResourcesToArraysL(TInt& aError, TInt& aTextId)
sl@0
   358
    {
sl@0
   359
    ASSERT(!iCoe);
sl@0
   360
    TResourceReader reader;
sl@0
   361
    ReadLocalizedSeparatorCharacterFromResourceAndPrepareResourceReaderLC(reader);
sl@0
   362
sl@0
   363
    // Lets handle KErrNoMemory as a special case, this this way we can maybe avoid extra "Resolving Errors"
sl@0
   364
    if (aError == KErrNoMemory) 
sl@0
   365
        {
sl@0
   366
        ASSERT(!iTextBuffer && !iTitleText);
sl@0
   367
		aTextId = R_ERROR_RES_NO_MEMORY;
sl@0
   368
		iTextBuffer = AllocReadUnicodeString(iResFile, aTextId);
sl@0
   369
		iTitleText = AllocReadUnicodeString(iResFile, R_BASE);
sl@0
   370
        User::Leave(KErrNoMemory);
sl@0
   371
        }
sl@0
   372
   
sl@0
   373
	if(iStartError && iErrorTexts && iFlags && iAppTexts)
sl@0
   374
		{
sl@0
   375
		CleanupStack::PopAndDestroy(); // rBuffer
sl@0
   376
		return;	// Already all done
sl@0
   377
		}
sl@0
   378
        
sl@0
   379
	// The resource data is used differenciate between the old and new resource formats 
sl@0
   380
	// (i.e, resources using title text and one's without using title text)
sl@0
   381
	// This API only support the new format, so just check that it is on the new format
sl@0
   382
    const TInt dummy = reader.ReadInt16();
sl@0
   383
    const TInt version = reader.ReadInt16();
sl@0
   384
    __ASSERT_ALWAYS(dummy == 0 && version == 2, User::Leave(KErrNotSupported));
sl@0
   385
sl@0
   386
    // Read into the error arrays
sl@0
   387
    const TInt arrayCount = reader.ReadInt16();
sl@0
   388
    
sl@0
   389
    ASSERT(!iStartError && !iErrorTexts && !iFlags && !iAppTexts);
sl@0
   390
    iStartError = new(ELeave) CArrayFixFlat<TInt>(arrayCount);
sl@0
   391
    iErrorTexts = new(ELeave) CArrayPtrSeg<CErrorResourceIdArray>(arrayCount);
sl@0
   392
    iFlags = new(ELeave) CArrayPtrSeg<CErrorResourceFlagArray>(arrayCount);
sl@0
   393
    iAppTexts = new(ELeave) CArrayFixFlat<TInt>(arrayCount);
sl@0
   394
    
sl@0
   395
    for (TInt cc = 0; cc < arrayCount; ++cc)
sl@0
   396
        {
sl@0
   397
        // Read in error array
sl@0
   398
        iAppTexts->AppendL(reader.ReadInt32()); // Application indicators  
sl@0
   399
        iStartError->AppendL(reader.ReadInt32());
sl@0
   400
sl@0
   401
        const TInt listId = reader.ReadInt32();
sl@0
   402
        
sl@0
   403
        TResourceReader reader2;
sl@0
   404
        HBufC8* rBuffer = iResFile.AllocReadL(listId & KRIMask); // Remove offset from id
sl@0
   405
        reader2.SetBuffer(rBuffer);
sl@0
   406
        CleanupStack::PushL(rBuffer);
sl@0
   407
sl@0
   408
        const TInt count = reader2.ReadInt16();
sl@0
   409
        CArrayFixFlat<TInt>* array = new(ELeave) CErrorResourceIdArray(count);
sl@0
   410
        CleanupStack::PushL(array);
sl@0
   411
        
sl@0
   412
        CArrayFixFlat<TInt>* flagarray = new(ELeave) CErrorResourceFlagArray(count);
sl@0
   413
        CleanupStack::PushL(flagarray);
sl@0
   414
    
sl@0
   415
        for (TInt dd = 0; dd < count; ++dd)
sl@0
   416
            {                   
sl@0
   417
            const TInt flags = reader2.ReadInt8();
sl@0
   418
            flagarray->AppendL(flags);
sl@0
   419
            // Append resource id for this error text
sl@0
   420
            array->AppendL(reader2.ReadInt32());
sl@0
   421
            }
sl@0
   422
    
sl@0
   423
        iFlags->AppendL(flagarray);
sl@0
   424
        CleanupStack::Pop(flagarray); 
sl@0
   425
sl@0
   426
        iErrorTexts->AppendL(array);
sl@0
   427
        CleanupStack::Pop(array);
sl@0
   428
sl@0
   429
        CleanupStack::PopAndDestroy(rBuffer);
sl@0
   430
        }
sl@0
   431
    
sl@0
   432
    CleanupStack::PopAndDestroy(); // reader's rBuffer
sl@0
   433
    }
sl@0
   434
sl@0
   435
// ---------------------------------------------------------
sl@0
   436
// CTextResolver::Reset()
sl@0
   437
//
sl@0
   438
// Reset the member variables
sl@0
   439
// ---------------------------------------------------------
sl@0
   440
//
sl@0
   441
void CTextResolver::Reset()
sl@0
   442
    {
sl@0
   443
    if (iBaseResourceFileOffset)
sl@0
   444
        {
sl@0
   445
        iCoe->DeleteResourceFile(iBaseResourceFileOffset);
sl@0
   446
        iBaseResourceFileOffset = 0;
sl@0
   447
        }
sl@0
   448
        
sl@0
   449
    if (iErrorTexts)
sl@0
   450
        {
sl@0
   451
        iErrorTexts->ResetAndDestroy();
sl@0
   452
        delete iErrorTexts;
sl@0
   453
        iErrorTexts = NULL;
sl@0
   454
        }
sl@0
   455
    
sl@0
   456
    if (iFlags)
sl@0
   457
        {
sl@0
   458
        iFlags->ResetAndDestroy();
sl@0
   459
        delete iFlags;
sl@0
   460
        iFlags = NULL;
sl@0
   461
        }
sl@0
   462
                
sl@0
   463
    delete iStartError;
sl@0
   464
    iStartError = NULL;
sl@0
   465
    
sl@0
   466
    delete iAppTexts;   
sl@0
   467
    iAppTexts = NULL;
sl@0
   468
    }
sl@0
   469
sl@0
   470
/**
sl@0
   471
ErrorResolver resource files in the ?:/resource/errors/ folder must always begin with
sl@0
   472
an array of error texts. The Nokia-specific version of the errorres.rss file also 
sl@0
   473
contains the localized titel/text separator character at the end of the file.
sl@0
   474
This method reads the separator character from file using the CCoeEnv provided.
sl@0
   475
*/
sl@0
   476
void CTextResolver::ReadLocalizedSeparatorCharacterFromResourceL(CCoeEnv& aCoeEnv)
sl@0
   477
	{
sl@0
   478
	ASSERT(&aCoeEnv);
sl@0
   479
	
sl@0
   480
	if (!iBaseResourceFileOffset)
sl@0
   481
    	{
sl@0
   482
    	TFileName filename(KErrorResResourceFileName);
sl@0
   483
    	BaflUtils::NearestLanguageFile(aCoeEnv.FsSession(), filename);
sl@0
   484
    
sl@0
   485
    	iBaseResourceFileOffset = aCoeEnv.AddResourceFileL(filename);
sl@0
   486
		
sl@0
   487
		ASSERT(!iContextSeparator);
sl@0
   488
		iContextSeparator = aCoeEnv.AllocReadResourceL(R_LOCALIZED_CONTEXT_SEPARATOR);
sl@0
   489
		}
sl@0
   490
	}
sl@0
   491
sl@0
   492
/**
sl@0
   493
ErrorResolver resource files in the ?:/resource/errors/ folder must always begin with
sl@0
   494
an array of error texts. The Nokia-specific version of the errorres.rss file also 
sl@0
   495
contains the localized titel/text separator character at the end of the file.
sl@0
   496
This method reads the separator character from file WITHOUT using a CCoeEnv.
sl@0
   497
*/
sl@0
   498
void CTextResolver::ReadLocalizedSeparatorCharacterFromResourceAndPrepareResourceReaderLC(TResourceReader& aResReader)
sl@0
   499
    {
sl@0
   500
    ASSERT(!iCoe);
sl@0
   501
    ASSERT(!iBaseResourceFileOffset);
sl@0
   502
sl@0
   503
    HBufC8* rBuffer = iResFile.AllocReadL(R_ERROR_RES_BASE_LIST & KRIMask); // Remove offset from id
sl@0
   504
    aResReader.SetBuffer(rBuffer);        
sl@0
   505
    CleanupStack::PushL(rBuffer);
sl@0
   506
    
sl@0
   507
    if(!iContextSeparator)
sl@0
   508
    	iContextSeparator = AllocReadUnicodeString(iResFile, R_LOCALIZED_CONTEXT_SEPARATOR);
sl@0
   509
    
sl@0
   510
    // Leave rBuffer on CleanupStack
sl@0
   511
    }
sl@0
   512
sl@0
   513
// returns NULL if out of memory
sl@0
   514
HBufC* CTextResolver::AllocReadUnicodeString(RResourceFile& aResFile, TInt aTextId)
sl@0
   515
    { 
sl@0
   516
    // Reading Unicode string w/o CCoe
sl@0
   517
    HBufC8* tempBuffer = NULL;
sl@0
   518
    HBufC* retBuffer = NULL;
sl@0
   519
sl@0
   520
    TRAPD(err, tempBuffer = aResFile.AllocReadL((KRIMask & aTextId))); // Remove offset from id
sl@0
   521
    if (!err)
sl@0
   522
        {
sl@0
   523
        __ASSERT_DEBUG((tempBuffer->Length()%2)==0,Panic(aTextId));
sl@0
   524
sl@0
   525
        // do bitwise shift to halve the length for mapping the tempBuffer to unicode.
sl@0
   526
        TInt newLen = tempBuffer->Length()>>1;
sl@0
   527
        TPtrC tempPointer((TText*)tempBuffer->Ptr(), newLen);
sl@0
   528
sl@0
   529
        retBuffer = tempPointer.Alloc();
sl@0
   530
        delete tempBuffer;
sl@0
   531
        }
sl@0
   532
sl@0
   533
    return retBuffer;
sl@0
   534
    }
sl@0
   535
sl@0
   536
TBool CTextResolver::AddContextAndSeparator(TErrorContext aContext)
sl@0
   537
    {
sl@0
   538
    TBool retval(EFalse); 
sl@0
   539
    
sl@0
   540
    if (iTextBuffer && iTitleText && iContextSeparator) // If we got all the parts we need...
sl@0
   541
        {
sl@0
   542
        //... then put them together according to the aContext argument
sl@0
   543
        
sl@0
   544
        // Context checked after validity of string buffers is checked, 
sl@0
   545
        // because EFalse return value is required if any of those is NULL.
sl@0
   546
        if (aContext == ECtxNoCtxNoSeparator)
sl@0
   547
            retval = ETrue;
sl@0
   548
        else
sl@0
   549
            {
sl@0
   550
            HBufC* tempBuffer = iTextBuffer->ReAlloc(iTextBuffer->Length()+iContextSeparator->Length()+iTitleText->Length()+1);
sl@0
   551
sl@0
   552
            // If allocation succeeds, insert context. Otherwise iTextBuffer stays as it is.
sl@0
   553
            if (tempBuffer)
sl@0
   554
                {
sl@0
   555
                iTextBuffer = tempBuffer;
sl@0
   556
				
sl@0
   557
				// Add the title ("context") text and separator before the error text
sl@0
   558
sl@0
   559
               	_LIT(KNewLineChar, "\n");
sl@0
   560
                iTextBuffer->Des().Insert(0, KNewLineChar);            
sl@0
   561
                iTextBuffer->Des().Insert(0, *iContextSeparator);
sl@0
   562
sl@0
   563
                if (aContext != ECtxNoCtx)
sl@0
   564
                    iTextBuffer->Des().Insert(0,*iTitleText);
sl@0
   565
sl@0
   566
                retval = ETrue;
sl@0
   567
                }
sl@0
   568
            }
sl@0
   569
        }
sl@0
   570
    return retval;
sl@0
   571
    }
sl@0
   572
sl@0
   573
void CTextResolver::AllocBuffersL()
sl@0
   574
	{
sl@0
   575
	delete iTitleText;
sl@0
   576
	iTitleText = NULL;
sl@0
   577
	delete iTextBuffer;
sl@0
   578
	iTextBuffer = NULL;
sl@0
   579
sl@0
   580
	iTitleText = HBufC::NewL(KEikErrorResolverMaxTextLength);
sl@0
   581
    iTextBuffer = HBufC::NewL(KEikErrorResolverMaxTextLength);
sl@0
   582
	}
sl@0
   583
	
sl@0
   584
void CTextResolver::DoResolveErrorStringL(TInt aError, TInt& aTextId, TUint& aFlags)
sl@0
   585
	{
sl@0
   586
	ASSERT(iCoe);
sl@0
   587
	AllocBuffersL();
sl@0
   588
    
sl@0
   589
   	TPtr errTextPtr = iTextBuffer->Des();
sl@0
   590
	TPtr errTitlePtr = iTitleText->Des();
sl@0
   591
    
sl@0
   592
    CEikonEnv::TErrorValidity errValidity = static_cast<CEikonEnv*>(iCoe)->DoGetErrorTextAndTitle(errTextPtr, aError, aTextId, aFlags, errTitlePtr, ETrue);
sl@0
   593
sl@0
   594
    if (errValidity == CEikonEnv::EErrorNumInvalid)
sl@0
   595
    	Panic(aError);
sl@0
   596
sl@0
   597
	// If either the iTextBuffer or iTitleText buffers were too small, 
sl@0
   598
	// they will contain the required length coded as digits into the buffer
sl@0
   599
	TLex sizeRealloc(*iTextBuffer);
sl@0
   600
    TInt sizeRequiredForTextBuffer = 0;
sl@0
   601
    TInt sizeRequiredForTitleBuffer = 0;
sl@0
   602
    
sl@0
   603
    // Try reading the size required, if any
sl@0
   604
    sizeRealloc.Val(sizeRequiredForTextBuffer);
sl@0
   605
    sizeRealloc = *iTitleText;
sl@0
   606
    sizeRealloc.Val(sizeRequiredForTitleBuffer);
sl@0
   607
    
sl@0
   608
    // If sizes were found, realloc and get the error text again
sl@0
   609
    if (sizeRequiredForTextBuffer || sizeRequiredForTitleBuffer)	
sl@0
   610
    	{	
sl@0
   611
		if (sizeRequiredForTextBuffer)
sl@0
   612
			{
sl@0
   613
			delete iTextBuffer;
sl@0
   614
			iTextBuffer = NULL;
sl@0
   615
			iTextBuffer = HBufC::NewL(sizeRequiredForTextBuffer);
sl@0
   616
			}
sl@0
   617
			
sl@0
   618
		if (sizeRequiredForTitleBuffer)
sl@0
   619
			{
sl@0
   620
			delete iTitleText;
sl@0
   621
			iTitleText = NULL;
sl@0
   622
			iTitleText = HBufC::NewL(sizeRequiredForTitleBuffer);
sl@0
   623
			}
sl@0
   624
sl@0
   625
	   	TPtr errTextPtr(iTextBuffer->Des());
sl@0
   626
		TPtr errTitlePtr(iTitleText->Des());
sl@0
   627
		errValidity = static_cast<CEikonEnv*>(iCoe)->DoGetErrorTextAndTitle(errTextPtr, aError, aTextId, aFlags, errTitlePtr, ETrue);
sl@0
   628
		if (!iTextBuffer->Length())
sl@0
   629
			{
sl@0
   630
			delete iTextBuffer;
sl@0
   631
			iTextBuffer = NULL;
sl@0
   632
			}
sl@0
   633
    	}
sl@0
   634
    	
sl@0
   635
    ReadLocalizedSeparatorCharacterFromResourceL(*iCoe);
sl@0
   636
	}
sl@0
   637
sl@0
   638
//  End of File
sl@0
   639
sl@0
   640