os/ossrv/genericservices/s60compatibilityheaders/commonengine/src/textresolver.cpp
Update contrib.
1 // Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 #include <barsread.h> // TResourceReader
19 #include <bautils.h> // BaflUtils
20 #include "textresolver.h"
21 #include <errorres.rsg>
22 #include <uikon/eikdefconst.h>
25 const TInt KUnknownError( 0 );
26 const TInt KBlankError( -2 );
28 const TInt KRIMask(0x00000fff);
29 _LIT(KErrorResResourceFileName, "z:\\resource\\errors\\ERRORRES.RSC");
30 _LIT(KRDSupport, "c:\\resource\\ErrRD");
31 _LIT(KTextResolverPanic, "TextResolver");
34 GLDEF_C void Panic(TInt aPanic)
36 User::Panic(KTextResolverPanic, aPanic);
40 CTextResolver::CTextResolver()
44 CTextResolver::CTextResolver(CCoeEnv& aEnv)
49 * Two-phase constructor method that is used to create a new instance
50 * of the CTextResolver class. The implementation uses the passed
51 * CCoeEnv instance to access the resource files.
53 * @param aEnv the CCoeEnv instance to be used to access the resource
55 * @return a pointer to a new instance of the CTextResolver class.
57 EXPORT_C CTextResolver* CTextResolver::NewL(CCoeEnv& aEnv)
59 CTextResolver* self = CTextResolver::NewLC(aEnv);
60 CleanupStack::Pop(self);
65 * Constructor creates a new instance of CTextResolver. The
66 * implementation uses the passed CCoeEnv instance to access the
67 * resource files. Leaves the object on the cleanup stack.
69 * @param aEnv the CCoeEnv instance to be used to access the resource
71 * @return a pointer to a new instance of the CTextResolver class.
73 EXPORT_C CTextResolver* CTextResolver::NewLC(CCoeEnv& aEnv)
75 CTextResolver* self = new (ELeave) CTextResolver(aEnv);
76 CleanupStack::PushL(self);
82 * Constructor creates a new instance of CTextResolver. Resource files
83 * are accessed through the RResourceFile API.
85 * @return a pointer to a new instance of the CTextResolver class.
87 EXPORT_C CTextResolver* CTextResolver::NewL()
89 CTextResolver* self = CTextResolver::NewLC();
90 CleanupStack::Pop(self);
95 * Constructor creates a new instance of CTextResolver.Resource files
96 * are accessed through the RResourceFile API. Leaves the object on
99 * @return a pointer to a new instance of the CTextResolver class.
101 EXPORT_C CTextResolver* CTextResolver::NewLC()
103 CTextResolver* self = new (ELeave) CTextResolver();
104 CleanupStack::PushL(self);
110 // ---------------------------------------------------------
111 // CTextResolver::ConstructL
113 // Symbian OS default constructor, initializes variables and cache
114 // ---------------------------------------------------------
116 void CTextResolver::ConstructL()
118 // if no coe is present, connect new filesession.
121 User::LeaveIfError(iFs.Connect());
123 TFileName resFile(KErrorResResourceFileName);
124 BaflUtils::NearestLanguageFile(iFs, resFile);
125 iResFile.OpenL(iFs, resFile);
128 iRDSupport = BaflUtils::FileExists(iCoe ? iCoe->FsSession() : iFs, KRDSupport);
134 EXPORT_C CTextResolver::~CTextResolver()
139 // if no coe is present, close filesession.
145 delete iContextSeparator;
149 * Resolves the given error code and returns the error text for the
150 * resolved error. Resolved text can be of any length. This version
151 * is for "normal" use.
153 * @param aError The error code to be resolved.
154 * @param aContext Optional context for error numbers. If the aContext
155 * parameter is not passed to the function, it uses the default value
157 * @return the error text for the resolved error. "System error" (in
158 * English localisation) is returned when error code is not known. For
159 * unknown errors blank error flag (flags are defined in
160 * textresolver.hrh) is also set to hide errors without proper
161 * explanation. There is no limit on how long the resolved string
164 EXPORT_C const TDesC& CTextResolver::ResolveErrorString(TInt aError, TErrorContext aContext)
168 return ResolveErrorString(aError,id,flags,aContext);
172 * Resolves the given error code and returns the error text for the
173 * resolved error. Resolved text can be of any length. This version
174 * is for advanced use
176 * @param aError The error code to be resolved.
177 * @param aTextId ID of the returned text.
178 * @param aFlags The priority of the returned error. The priority is
179 * defined by the this module! Flags are defined in textresolver.hrh.
180 * @param aContext Optional context for error numbers. If the aContext
181 * parameter is not passed to the function, it uses the default value
183 * @return the error text for the resolved error. "System error" (in
184 * English localisation) is returned when error code is not known. For
185 * unknown errors blank error flag (flags are defined in
186 * textresolver.hrh) is also set to hide errors without proper
187 * explanation. There is no limit on how long the resolved string
190 EXPORT_C const TDesC& CTextResolver::ResolveErrorString(TInt aError, TInt& aTextId, TUint& aFlags, TErrorContext aContext)
193 if (iCoe) // Use the resource loaded using the CCoeEnv to read the resource file
195 TRAPD(err, DoResolveErrorStringL(aError, aTextId, aFlags));
198 if ((!AddContextAndSeparator(aContext)))
200 // Even the resource file opening failed, return OOM flag
201 aFlags |= EErrorResOOMFlag;
207 if(aFlags&ETextResolverUnknownErrorFlag) // Error code not supported. No error text found.
209 // Error is not recognised
210 aTextId = R_ERROR_RES_GENERAL; // Read and return an "unknown error" text from the
211 // Nokia-specific version of the errorres.rss file
212 TPtr appTextPtr = iTitleText->Des();
213 TPtr textPtr = iTextBuffer->Des();
214 iCoe->ReadResourceL(appTextPtr, R_BASE);
215 iCoe->ReadResourceL(textPtr, R_ERROR_RES_GENERAL);
218 else // No CCoeEnv available
221 TRAPD(err, DoRawReadOfSystemErrorResourcesToArraysL(aError, aTextId));
224 // Cleaning needed since object not destroyed
227 if (aTextId != R_ERROR_RES_NO_MEMORY || !AddContextAndSeparator(aContext))
229 // Even the resource file opening failed, return OOM flag
230 aFlags |= EErrorResOOMFlag;
237 aTextId = ResourceForError(aError);
242 aFlags |= ETextResolverBlankErrorFlag;
247 // Error is not recognised
248 aTextId = R_ERROR_RES_GENERAL;
251 iTitleText = AllocReadUnicodeString(iResFile, R_BASE);
252 aFlags |= ETextResolverUnknownErrorFlag;
253 // No break here; Fall through to default case to resolve text for the general error
257 // --- Error recognised by the resolver, so get the text ---
259 iTextBuffer = AllocReadUnicodeString(iResFile, aTextId);
265 if(aFlags&ETextResolverBlankErrorFlag) // Error code supported, but no error text found.
268 iTextBuffer = NULL; // Return no error text at all
271 if (!AddContextAndSeparator(aContext))
273 aFlags |= EErrorResOOMFlag;
277 if (iRDSupport && iTextBuffer)
279 HBufC* tempBuffer = iTextBuffer->ReAlloc(iTextBuffer->Length()+14);
281 // if allocation succeeds, append error id. Otherwise iTextBuffer stays as it is
284 iTextBuffer = tempBuffer;
285 iTextBuffer->Des().Append(' ');
286 iTextBuffer->Des().Append('(');
287 iTextBuffer->Des().AppendNum(aError);
288 iTextBuffer->Des().Append(')');
291 else if (aFlags & ETextResolverUnknownErrorFlag)
292 { // We hide the errors we cannot give proper explanation,
293 //i.e. "System Error" will not be shown.
294 aFlags |= ETextResolverBlankErrorFlag;
303 TInt CTextResolver::ResourceForError(TInt aError)
306 const TInt errorCount = iStartError->Count();
311 for (TInt cc = 0; cc < errorCount; ++cc)
313 const TInt starterror = (*iStartError)[cc];
314 const TInt endError = starterror-(*iErrorTexts)[cc]->Count()+1;
316 if (aError >= endError && aError <= starterror)
319 iTitleText = AllocReadUnicodeString(iResFile, (*iAppTexts)[cc]);
321 const TInt errorIndex = -(aError-starterror);
322 const TInt flags = (*iFlags)[cc]->At(errorIndex);
324 if (flags & ETextResolverPanicErrorFlag)
326 // --- Some errors are never meant to get to the UI level ---
327 // --- Those of them that do, result in a panic ---
331 else if (flags & ETextResolverBlankErrorFlag)
333 // --- Error code can be ignored from the users' perspective ---
334 // --- Return blank error text ---
338 else if (flags & ETextResolverUnknownErrorFlag)
340 // --- Error not recognised by TextResolver ---
341 // --- Will be passed on to UIKON for interpretation ---
342 return KUnknownError;
345 return (*iErrorTexts)[cc]->At(errorIndex);
349 return KUnknownError;
354 Read the system error texts from errorres.rss only, without using the Uikon CEikErrorResolver.
355 This will not pick up any appliation specific errors.
357 void CTextResolver::DoRawReadOfSystemErrorResourcesToArraysL(TInt& aError, TInt& aTextId)
360 TResourceReader reader;
361 ReadLocalizedSeparatorCharacterFromResourceAndPrepareResourceReaderLC(reader);
363 // Lets handle KErrNoMemory as a special case, this this way we can maybe avoid extra "Resolving Errors"
364 if (aError == KErrNoMemory)
366 ASSERT(!iTextBuffer && !iTitleText);
367 aTextId = R_ERROR_RES_NO_MEMORY;
368 iTextBuffer = AllocReadUnicodeString(iResFile, aTextId);
369 iTitleText = AllocReadUnicodeString(iResFile, R_BASE);
370 User::Leave(KErrNoMemory);
373 if(iStartError && iErrorTexts && iFlags && iAppTexts)
375 CleanupStack::PopAndDestroy(); // rBuffer
376 return; // Already all done
379 // The resource data is used differenciate between the old and new resource formats
380 // (i.e, resources using title text and one's without using title text)
381 // This API only support the new format, so just check that it is on the new format
382 const TInt dummy = reader.ReadInt16();
383 const TInt version = reader.ReadInt16();
384 __ASSERT_ALWAYS(dummy == 0 && version == 2, User::Leave(KErrNotSupported));
386 // Read into the error arrays
387 const TInt arrayCount = reader.ReadInt16();
389 ASSERT(!iStartError && !iErrorTexts && !iFlags && !iAppTexts);
390 iStartError = new(ELeave) CArrayFixFlat<TInt>(arrayCount);
391 iErrorTexts = new(ELeave) CArrayPtrSeg<CErrorResourceIdArray>(arrayCount);
392 iFlags = new(ELeave) CArrayPtrSeg<CErrorResourceFlagArray>(arrayCount);
393 iAppTexts = new(ELeave) CArrayFixFlat<TInt>(arrayCount);
395 for (TInt cc = 0; cc < arrayCount; ++cc)
397 // Read in error array
398 iAppTexts->AppendL(reader.ReadInt32()); // Application indicators
399 iStartError->AppendL(reader.ReadInt32());
401 const TInt listId = reader.ReadInt32();
403 TResourceReader reader2;
404 HBufC8* rBuffer = iResFile.AllocReadL(listId & KRIMask); // Remove offset from id
405 reader2.SetBuffer(rBuffer);
406 CleanupStack::PushL(rBuffer);
408 const TInt count = reader2.ReadInt16();
409 CArrayFixFlat<TInt>* array = new(ELeave) CErrorResourceIdArray(count);
410 CleanupStack::PushL(array);
412 CArrayFixFlat<TInt>* flagarray = new(ELeave) CErrorResourceFlagArray(count);
413 CleanupStack::PushL(flagarray);
415 for (TInt dd = 0; dd < count; ++dd)
417 const TInt flags = reader2.ReadInt8();
418 flagarray->AppendL(flags);
419 // Append resource id for this error text
420 array->AppendL(reader2.ReadInt32());
423 iFlags->AppendL(flagarray);
424 CleanupStack::Pop(flagarray);
426 iErrorTexts->AppendL(array);
427 CleanupStack::Pop(array);
429 CleanupStack::PopAndDestroy(rBuffer);
432 CleanupStack::PopAndDestroy(); // reader's rBuffer
435 // ---------------------------------------------------------
436 // CTextResolver::Reset()
438 // Reset the member variables
439 // ---------------------------------------------------------
441 void CTextResolver::Reset()
443 if (iBaseResourceFileOffset)
445 iCoe->DeleteResourceFile(iBaseResourceFileOffset);
446 iBaseResourceFileOffset = 0;
451 iErrorTexts->ResetAndDestroy();
458 iFlags->ResetAndDestroy();
471 ErrorResolver resource files in the ?:/resource/errors/ folder must always begin with
472 an array of error texts. The Nokia-specific version of the errorres.rss file also
473 contains the localized titel/text separator character at the end of the file.
474 This method reads the separator character from file using the CCoeEnv provided.
476 void CTextResolver::ReadLocalizedSeparatorCharacterFromResourceL(CCoeEnv& aCoeEnv)
480 if (!iBaseResourceFileOffset)
482 TFileName filename(KErrorResResourceFileName);
483 BaflUtils::NearestLanguageFile(aCoeEnv.FsSession(), filename);
485 iBaseResourceFileOffset = aCoeEnv.AddResourceFileL(filename);
487 ASSERT(!iContextSeparator);
488 iContextSeparator = aCoeEnv.AllocReadResourceL(R_LOCALIZED_CONTEXT_SEPARATOR);
493 ErrorResolver resource files in the ?:/resource/errors/ folder must always begin with
494 an array of error texts. The Nokia-specific version of the errorres.rss file also
495 contains the localized titel/text separator character at the end of the file.
496 This method reads the separator character from file WITHOUT using a CCoeEnv.
498 void CTextResolver::ReadLocalizedSeparatorCharacterFromResourceAndPrepareResourceReaderLC(TResourceReader& aResReader)
501 ASSERT(!iBaseResourceFileOffset);
503 HBufC8* rBuffer = iResFile.AllocReadL(R_ERROR_RES_BASE_LIST & KRIMask); // Remove offset from id
504 aResReader.SetBuffer(rBuffer);
505 CleanupStack::PushL(rBuffer);
507 if(!iContextSeparator)
508 iContextSeparator = AllocReadUnicodeString(iResFile, R_LOCALIZED_CONTEXT_SEPARATOR);
510 // Leave rBuffer on CleanupStack
513 // returns NULL if out of memory
514 HBufC* CTextResolver::AllocReadUnicodeString(RResourceFile& aResFile, TInt aTextId)
516 // Reading Unicode string w/o CCoe
517 HBufC8* tempBuffer = NULL;
518 HBufC* retBuffer = NULL;
520 TRAPD(err, tempBuffer = aResFile.AllocReadL((KRIMask & aTextId))); // Remove offset from id
523 __ASSERT_DEBUG((tempBuffer->Length()%2)==0,Panic(aTextId));
525 // do bitwise shift to halve the length for mapping the tempBuffer to unicode.
526 TInt newLen = tempBuffer->Length()>>1;
527 TPtrC tempPointer((TText*)tempBuffer->Ptr(), newLen);
529 retBuffer = tempPointer.Alloc();
536 TBool CTextResolver::AddContextAndSeparator(TErrorContext aContext)
538 TBool retval(EFalse);
540 if (iTextBuffer && iTitleText && iContextSeparator) // If we got all the parts we need...
542 //... then put them together according to the aContext argument
544 // Context checked after validity of string buffers is checked,
545 // because EFalse return value is required if any of those is NULL.
546 if (aContext == ECtxNoCtxNoSeparator)
550 HBufC* tempBuffer = iTextBuffer->ReAlloc(iTextBuffer->Length()+iContextSeparator->Length()+iTitleText->Length()+1);
552 // If allocation succeeds, insert context. Otherwise iTextBuffer stays as it is.
555 iTextBuffer = tempBuffer;
557 // Add the title ("context") text and separator before the error text
559 _LIT(KNewLineChar, "\n");
560 iTextBuffer->Des().Insert(0, KNewLineChar);
561 iTextBuffer->Des().Insert(0, *iContextSeparator);
563 if (aContext != ECtxNoCtx)
564 iTextBuffer->Des().Insert(0,*iTitleText);
573 void CTextResolver::AllocBuffersL()
580 iTitleText = HBufC::NewL(KEikErrorResolverMaxTextLength);
581 iTextBuffer = HBufC::NewL(KEikErrorResolverMaxTextLength);
584 void CTextResolver::DoResolveErrorStringL(TInt aError, TInt& aTextId, TUint& aFlags)
589 TPtr errTextPtr = iTextBuffer->Des();
590 TPtr errTitlePtr = iTitleText->Des();
592 CEikonEnv::TErrorValidity errValidity = static_cast<CEikonEnv*>(iCoe)->DoGetErrorTextAndTitle(errTextPtr, aError, aTextId, aFlags, errTitlePtr, ETrue);
594 if (errValidity == CEikonEnv::EErrorNumInvalid)
597 // If either the iTextBuffer or iTitleText buffers were too small,
598 // they will contain the required length coded as digits into the buffer
599 TLex sizeRealloc(*iTextBuffer);
600 TInt sizeRequiredForTextBuffer = 0;
601 TInt sizeRequiredForTitleBuffer = 0;
603 // Try reading the size required, if any
604 sizeRealloc.Val(sizeRequiredForTextBuffer);
605 sizeRealloc = *iTitleText;
606 sizeRealloc.Val(sizeRequiredForTitleBuffer);
608 // If sizes were found, realloc and get the error text again
609 if (sizeRequiredForTextBuffer || sizeRequiredForTitleBuffer)
611 if (sizeRequiredForTextBuffer)
615 iTextBuffer = HBufC::NewL(sizeRequiredForTextBuffer);
618 if (sizeRequiredForTitleBuffer)
622 iTitleText = HBufC::NewL(sizeRequiredForTitleBuffer);
625 TPtr errTextPtr(iTextBuffer->Des());
626 TPtr errTitlePtr(iTitleText->Des());
627 errValidity = static_cast<CEikonEnv*>(iCoe)->DoGetErrorTextAndTitle(errTextPtr, aError, aTextId, aFlags, errTitlePtr, ETrue);
628 if (!iTextBuffer->Length())
635 ReadLocalizedSeparatorCharacterFromResourceL(*iCoe);