os/ossrv/lowlevellibsandfws/apputils/src/BAUTILS.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) 1997-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
// Started by DWW, November 1995
sl@0
    15
// BAFL utilities static class
sl@0
    16
//
sl@0
    17
//
sl@0
    18
sl@0
    19
#include "BaUtilsImp.h"
sl@0
    20
#include <e32hal.h>
sl@0
    21
#include <bautils.h>
sl@0
    22
#include <baflpan.h>
sl@0
    23
#include <baliba.h>
sl@0
    24
#include <hal.h>
sl@0
    25
#include <hal_data.h>
sl@0
    26
#include <utf.h>
sl@0
    27
sl@0
    28
/**
sl@0
    29
Mimimum length of a filename and mimimum length of a suffix.
sl@0
    30
Note these two values are tied together.
sl@0
    31
*/
sl@0
    32
const TInt KInvNameAndMinSuffixLength = 2;
sl@0
    33
sl@0
    34
#define ISDIGIT(c) (c-'0' >= 0 && c-'0' <= 9)
sl@0
    35
sl@0
    36
_LIT(KAllDrives, "YXWVUTSRQPONMLKJIHGFEDCBAZ");
sl@0
    37
const TInt KDriveAndPathLength = 3;
sl@0
    38
sl@0
    39
// screen calibration stuff
sl@0
    40
_LIT(KScreenCalibrationFolder,"\\System\\Data\\");
sl@0
    41
_LIT(KScreenCalibrationFileName, "Screen.DAT");
sl@0
    42
const TInt KScreenCalibrationPathLength = 23;	// folder + filename
sl@0
    43
sl@0
    44
sl@0
    45
// #define DO_PROFILING
sl@0
    46
sl@0
    47
#if defined(DO_PROFILING)
sl@0
    48
#pragma message ("------------ N.B. profiling of \"BaflUtils::NearestLanguageFile\" is enabled")
sl@0
    49
#include <e32svr.h>
sl@0
    50
#define FIRST_PROFILE_INDEX	50
sl@0
    51
#define PROFILE_INDEX_1		(FIRST_PROFILE_INDEX+0)
sl@0
    52
#define PROFILE_COUNT		1
sl@0
    53
#endif
sl@0
    54
sl@0
    55
// BaflUtils
sl@0
    56
sl@0
    57
class BaflDir : public CDir
sl@0
    58
	{
sl@0
    59
public:
sl@0
    60
	void RemoveSystem();
sl@0
    61
	TInt SortByTable(CBaflFileSortTable* aTable);
sl@0
    62
private:
sl@0
    63
	TInt MinEntrySize(const TEntry & aEntry);
sl@0
    64
	};
sl@0
    65
sl@0
    66
sl@0
    67
LOCAL_C const TLanguage dp0[] = { ELangCanadianEnglish, ELangAmerican,ELangEnglish, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish,ELangNone };
sl@0
    68
LOCAL_C const TLanguage dp1[] = { ELangAmerican, ELangEnglish,ELangCanadianEnglish, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangNone };
sl@0
    69
LOCAL_C const TLanguage dp2[] = { ELangAustralian, ELangEnglish, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangCanadianEnglish,ELangNone };
sl@0
    70
LOCAL_C const TLanguage dp3[] = { ELangSouthAfricanEnglish, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangInternationalEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    71
LOCAL_C const TLanguage dp4[] = { ELangInternationalEnglish, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    72
LOCAL_C const TLanguage dp5[] = { ELangEnglish_Apac, ELangEnglish, ELangAustralian, ELangAmerican,ELangInternationalEnglish,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    73
LOCAL_C const TLanguage dp6[] = { ELangEnglish_Taiwan, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangInternationalEnglish,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    74
LOCAL_C const TLanguage dp7[] = { ELangEnglish_HongKong, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangInternationalEnglish,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    75
LOCAL_C const TLanguage dp8[] = { ELangEnglish_Prc, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangInternationalEnglish,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    76
LOCAL_C const TLanguage dp9[] = { ELangEnglish_Japan, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangInternationalEnglish,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    77
LOCAL_C const TLanguage dp10[] = { ELangEnglish_Thailand, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangInternationalEnglish,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    78
LOCAL_C const TLanguage dp11[] = { ELangEnglish_India, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangInternationalEnglish,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone };
sl@0
    79
LOCAL_C const TLanguage dp12[] = { ELangNewZealand, ELangEnglish, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangCanadianEnglish,ELangNone };
sl@0
    80
LOCAL_C const TLanguage dp13[] = { ELangInternationalFrench,ELangFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
sl@0
    81
LOCAL_C const TLanguage dp14[] = { ELangBelgianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangCanadianFrench,ELangNone };
sl@0
    82
LOCAL_C const TLanguage dp15[] = { ELangCanadianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangNone };
sl@0
    83
LOCAL_C const TLanguage dp16[] = { ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
sl@0
    84
LOCAL_C const TLanguage dp17[] = { ELangSwissFrench,ELangFrench,ELangInternationalFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
sl@0
    85
LOCAL_C const TLanguage dp18[] = { ELangSwissGerman,ELangGerman,ELangAustrian,ELangNone };
sl@0
    86
LOCAL_C const TLanguage dp19[] = { ELangAustrian,ELangGerman,ELangSwissGerman,ELangNone };
sl@0
    87
LOCAL_C const TLanguage dp20[] = { ELangGerman,ELangSwissGerman,ELangAustrian,ELangNone };
sl@0
    88
LOCAL_C const TLanguage dp21[] = { ELangSerbian,ELangCroatian,ELangNone };
sl@0
    89
LOCAL_C const TLanguage dp22[] = { ELangCroatian,ELangSerbian,ELangNone };
sl@0
    90
LOCAL_C const TLanguage dp23[] = { ELangRomanian,ELangMoldavian,ELangNone };
sl@0
    91
LOCAL_C const TLanguage dp24[] = { ELangMoldavian,ELangRomanian,ELangNone };
sl@0
    92
LOCAL_C const TLanguage dp25[] = { ELangBelgianFlemish,ELangDutch,ELangNone };
sl@0
    93
LOCAL_C const TLanguage dp26[] = { ELangDutch,ELangBelgianFlemish,ELangNone };
sl@0
    94
LOCAL_C const TLanguage dp27[] = { ELangAfrikaans,ELangDutch,ELangNone };
sl@0
    95
LOCAL_C const TLanguage dp28[] = { ELangMalay_Apac,ELangMalay,ELangNone };
sl@0
    96
LOCAL_C const TLanguage dp29[] = { ELangIndonesian_Apac,ELangIndonesian,ELangNone };
sl@0
    97
LOCAL_C const TLanguage dp30[] = { ELangSpanish,ELangInternationalSpanish,ELangLatinAmericanSpanish,ELangNone };
sl@0
    98
LOCAL_C const TLanguage dp31[] = { ELangLatinAmericanSpanish,ELangSpanish,ELangInternationalSpanish,ELangNone };
sl@0
    99
LOCAL_C const TLanguage dp32[] = { ELangInternationalSpanish,ELangSpanish,ELangLatinAmericanSpanish,ELangNone };
sl@0
   100
LOCAL_C const TLanguage dp33[] = { ELangCyprusGreek,ELangGreek,ELangNone };
sl@0
   101
LOCAL_C const TLanguage dp34[] = { ELangGreek,ELangCyprusGreek,ELangNone };
sl@0
   102
LOCAL_C const TLanguage dp35[] = { ELangSwissItalian,ELangItalian,ELangNone };
sl@0
   103
LOCAL_C const TLanguage dp36[] = { ELangItalian,ELangSwissItalian,ELangNone };
sl@0
   104
LOCAL_C const TLanguage dp37[] = { ELangBrazilianPortuguese,ELangPortuguese,ELangNone };
sl@0
   105
LOCAL_C const TLanguage dp38[] = { ELangPortuguese,ELangBrazilianPortuguese,ELangNone };
sl@0
   106
LOCAL_C const TLanguage dp39[] = { ELangFinlandSwedish,ELangSwedish,ELangNone };
sl@0
   107
LOCAL_C const TLanguage dp40[] = { ELangSwedish,ELangFinlandSwedish,ELangNone };
sl@0
   108
LOCAL_C const TLanguage dp41[] = { ELangCyprusTurkish,ELangTurkish,ELangNone };
sl@0
   109
LOCAL_C const TLanguage dp42[] = { ELangTurkish,ELangCyprusTurkish,ELangNone };
sl@0
   110
LOCAL_C const TLanguage dp43[] = { ELangHongKongChinese, ELangTaiwanChinese, ELangPrcChinese,ELangNone };
sl@0
   111
LOCAL_C const TLanguage dp44[] = { ELangTaiwanChinese, ELangHongKongChinese,ELangPrcChinese,ELangNone };
sl@0
   112
LOCAL_C const TLanguage dp45[] = { ELangPrcChinese, ELangHongKongChinese, ELangTaiwanChinese,ELangNone };
sl@0
   113
LOCAL_C const TLanguage * const KEquivalentLists[] = { dp0,  dp1,  dp2,  dp3,  dp4,  dp5,  dp6,  
sl@0
   114
		dp7,  dp8,  dp9,  dp10,  dp11,  dp12,  dp13,  dp14,  dp15,  dp16,  dp17,
sl@0
   115
		dp18,  dp19,  dp20,  dp21,  dp22,  dp23,  dp24,  dp25,  dp26,  dp27,  
sl@0
   116
		dp28,  dp29,  dp30,  dp31,  dp32,  dp33,  dp34,  dp35,  dp36,  dp37,  
sl@0
   117
		dp38,  dp39,  dp40,  dp41,  dp42,  dp43,  dp44,  dp45};
sl@0
   118
sl@0
   119
/**
sl@0
   120
This function gets the list of languages that are 'equivalent' to the
sl@0
   121
given language. We say language L1 is equivalent to language L2 if 
sl@0
   122
speakers of L2 can readily understand L1 without intentional study 
sl@0
   123
or extraordinary effort.
sl@0
   124
sl@0
   125
The equivalence relationship is defined in a static table. Please refer 
sl@0
   126
to the definition of 'KEquivalentLists' for details.
sl@0
   127
Each row in the table is formatted like this:
sl@0
   128
@code
sl@0
   129
L1, L2, L3, ..., Ln, ELangNone
sl@0
   130
@codeend
sl@0
   131
In which L2, ..., Ln are equivalents of L1, and ELangNone marks the end of an
sl@0
   132
entry. The list is ordered. Compared with L3, L2 is nearer to L1. When choosing
sl@0
   133
an equivalent of L1, L2 shall be preferred over L3, L3 shall be preferred 
sl@0
   134
over L4, and so on.  
sl@0
   135
L1 is always returned as the ‘nearest equivalent’ of L1 itself.
sl@0
   136
sl@0
   137
BaflUtils::NearestLanguageFileV2 searches language specific resource files 
sl@0
   138
according to the 'equivalent' relationship returned by this function.
sl@0
   139
 
sl@0
   140
@param aLang The language whose equivalents needs to be found out.
sl@0
   141
@param aEquivalents On return, this array contains the ordered list of 
sl@0
   142
       languages that are equivalent to the given language. If there is no 
sl@0
   143
       entry for the given language in the table, this array will contain 
sl@0
   144
       two elements on return: the first is the given language itself 
sl@0
   145
       and the 2nd one is ELangNone. For any language that has equivalents 
sl@0
   146
       defined, content of he corresponding entry is returned.     
sl@0
   147
       
sl@0
   148
@see BaflUtils::NearestLanguageFileV2
sl@0
   149
*/ 
sl@0
   150
EXPORT_C void 
sl@0
   151
BaflUtils::GetEquivalentLanguageList(TLanguage aLang, TLanguagePath& aEquivalents) 
sl@0
   152
	{
sl@0
   153
	aEquivalents[0] = aLang;
sl@0
   154
	aEquivalents[1] = ELangNone;
sl@0
   155
	const TInt len = sizeof(KEquivalentLists) / sizeof(KEquivalentLists[0]);
sl@0
   156
	for (TInt i = 0; i < len; ++i)
sl@0
   157
		{
sl@0
   158
		const TLanguage *ptr = KEquivalentLists[i];
sl@0
   159
		if (ptr[0] == aLang)
sl@0
   160
			{
sl@0
   161
			TInt index = 1;
sl@0
   162
			while (ELangNone != *ptr)
sl@0
   163
				{
sl@0
   164
				aEquivalents[index++] = (TLanguage)*(++ptr);
sl@0
   165
				}
sl@0
   166
			aEquivalents[index] = ELangNone;
sl@0
   167
			break;
sl@0
   168
			} // end if ptr[0]
sl@0
   169
		} // end for i
sl@0
   170
	}
sl@0
   171
sl@0
   172
/**
sl@0
   173
NearestLanguageFileV2 is very similar to the existing 'NearestLanguageFile'
sl@0
   174
function. The only difference between NearestLanguageFile and
sl@0
   175
NearestLanguageFileV2 is the order in which language specific 
sl@0
   176
resource files are searched for. 
sl@0
   177
NearestLanguageFile searches language specific resource files in the 
sl@0
   178
order defined by the 'downgrade path' of the given language. Content of the 
sl@0
   179
downgrade path is dependent on the current active locale, and parts of 
sl@0
   180
it is runtime configurable.
sl@0
   181
NearestLanguageFileV2 searches for language specific resource files 
sl@0
   182
in the order defined by the 'language equivalence table', which is a  
sl@0
   183
static data table fixed at build time. There is one entry in the table for 
sl@0
   184
each language that has one or more equivalents.
sl@0
   185
sl@0
   186
@param aFs An active file server session.
sl@0
   187
@param aName Name of the language-neutral resource file name which consist of
sl@0
   188
an optional drive specification, followed by an optional path name,
sl@0
   189
followed by basename for filename, followed by a period and extension.
sl@0
   190
On return, in case of a match, this is replaced by the language-specific version
sl@0
   191
which consists of the last two characters of the extension plus any preceding
sl@0
   192
numeric characters being replaced by the language code. Remains unchanged when there's no match 
sl@0
   193
@param aLanguage On return, in case of a match, this is replaced by the corresponding language.
sl@0
   194
  In case of no match, it is set to ELangNone.  
sl@0
   195
sl@0
   196
@see TLanguage
sl@0
   197
@see BaflUtils::GetEquivalentLanguageList
sl@0
   198
 */
sl@0
   199
EXPORT_C void 
sl@0
   200
BaflUtils::NearestLanguageFileV2(const RFs& aFs,TFileName& aName, TLanguage& aLanguage)
sl@0
   201
	{
sl@0
   202
	TNearestLanguageFileFinder finder(aFs);
sl@0
   203
	TBool goodSuffix=finder.SetFileName(aName);
sl@0
   204
	
sl@0
   205
	// Continue only if the suffix is good.
sl@0
   206
	if(goodSuffix)
sl@0
   207
		{
sl@0
   208
		// add preset customised resource drive to drive list  
sl@0
   209
		// Note that errors returned from AddCustomResourceDrive are ignored. This is because if 
sl@0
   210
		// a custom resource drive has not been found we still want to continue on with searching 
sl@0
   211
		// other drives according to our algorithm
sl@0
   212
		finder.AddCustomResourceDrive();
sl@0
   213
		
sl@0
   214
		GetEquivalentLanguageList(User::Language(), finder.iPath);
sl@0
   215
		if (!finder.FindLanguageAndDrive()
sl@0
   216
			&& KErrNone != finder.FindFirstLanguageFileAndDrive())
sl@0
   217
			finder.RepairFileName();
sl@0
   218
		aLanguage = finder.Language();
sl@0
   219
		}
sl@0
   220
	else
sl@0
   221
		{
sl@0
   222
		aLanguage = ELangNone;
sl@0
   223
		}
sl@0
   224
	}
sl@0
   225
sl@0
   226
// TLibAssocBase
sl@0
   227
sl@0
   228
EXPORT_C TLibAssocBase::TLibAssocBase(const RLibrary& aLib,TAny* aPtr)
sl@0
   229
	: iLibrary(aLib),iPtr(aPtr)
sl@0
   230
/**
sl@0
   231
Constructs the object taking the specified DLL and a class instance.
sl@0
   232
sl@0
   233
@param aLib A reference to a DLL that has already been opened.
sl@0
   234
@param aPtr An untyped pointer to an object to be associated with the DLL.
sl@0
   235
            Typically, this object will have been created using
sl@0
   236
            the ordinal 1 function from that DLL. */	
sl@0
   237
	{}
sl@0
   238
sl@0
   239
sl@0
   240
sl@0
   241
sl@0
   242
EXPORT_C void TLibAssocBase::Set(const RLibrary& aLib,TAny* aPtr)
sl@0
   243
/**
sl@0
   244
Implements TLibAssoc::Set().
sl@0
   245
sl@0
   246
@param aLib   A reference to a DLL that has already been opened.
sl@0
   247
@param aClass A pointer to an object to be associated with the DLL.
sl@0
   248
              Typically, this object will have been created using
sl@0
   249
              the ordinal 1 function from that DLL.
sl@0
   250
sl@0
   251
@see TLibAssoc::Set */
sl@0
   252
	{
sl@0
   253
	__ASSERT_ALWAYS(iLibrary.Handle()==KNullHandle&&iPtr==NULL,Panic(EBafPanicLibAssocAlreadySet));
sl@0
   254
	iLibrary=aLib;
sl@0
   255
	iPtr=aPtr;
sl@0
   256
	}
sl@0
   257
sl@0
   258
sl@0
   259
sl@0
   260
sl@0
   261
EXPORT_C void TLibAssocBase::DoUnload(TAny* aThis)
sl@0
   262
/**
sl@0
   263
Calls Close() on the associated DLL.
sl@0
   264
sl@0
   265
@param aThis An untyped pointer to a TLibAssoc type.
sl@0
   266
*/
sl@0
   267
	{
sl@0
   268
	TLibAssocBase& l=*(TLibAssocBase*)aThis;
sl@0
   269
	l.iPtr=NULL;
sl@0
   270
	l.iLibrary.Close();
sl@0
   271
	}
sl@0
   272
sl@0
   273
//
sl@0
   274
// class BaflUtils
sl@0
   275
//
sl@0
   276
EXPORT_C void BaflUtils::CopyWithTruncation(TDes& aDest,const TDesC& aSrc,TChar aTruncationSymbol)
sl@0
   277
/** Copies a descriptor, abbreviating it to fit the destination descriptor.
sl@0
   278
sl@0
   279
If aSrc is less than the maximum length of aDest, then the string is simply 
sl@0
   280
copied to aDest.
sl@0
   281
sl@0
   282
If this is not so, then the left-most characters of aSrc are copied to aDest, 
sl@0
   283
up to aDest's maximum length-1. aDest's final character is set to be aTruncationSymbol.
sl@0
   284
sl@0
   285
@param aDest On return, the truncated string
sl@0
   286
@param aSrc The string to truncate
sl@0
   287
@param aTruncationSymbol The truncation character to add */
sl@0
   288
	{ // static
sl@0
   289
	TInt maxLength=aDest.MaxLength();
sl@0
   290
	if (aSrc.Length()<=maxLength)
sl@0
   291
		aDest.Copy(aSrc);
sl@0
   292
	else
sl@0
   293
		{
sl@0
   294
		aDest.Copy(aSrc.Left(maxLength-1));
sl@0
   295
		aDest.Append(aTruncationSymbol);
sl@0
   296
		}
sl@0
   297
	}
sl@0
   298
sl@0
   299
EXPORT_C TBool BaflUtils::FileExists(const RFs& aFileSession,const TDesC& aFileName)
sl@0
   300
/** Checks if the specified file exists.
sl@0
   301
	
sl@0
   302
@param aFs File server session
sl@0
   303
@param aFileName File to check
sl@0
   304
@return ETrue if the file exists, otherwise EFalse */
sl@0
   305
	{ // static
sl@0
   306
	TEntry entry;
sl@0
   307
	return(aFileSession.Entry(aFileName,entry)==KErrNone);
sl@0
   308
	}
sl@0
   309
sl@0
   310
EXPORT_C TBool BaflUtils::PathExists(RFs& aFs,const TDesC& aPath)
sl@0
   311
/** Tests whether a path exists.
sl@0
   312
sl@0
   313
The path should contain a drive letter and a directory, or EFalse is returned. 
sl@0
   314
EFalse is also returned if it contains a filename or filename extension.
sl@0
   315
sl@0
   316
If the path is badly formed, for instance if it contains illegal characters, 
sl@0
   317
or any directory name consists of a single or double dot, or any directory 
sl@0
   318
name includes wildcard characters, the function returns EFalse.
sl@0
   319
sl@0
   320
@param aFs A connected session with the file server.
sl@0
   321
@param aPath The path to test for. It should end in a backslash.
sl@0
   322
@return ETrue if the path exists, EFalse if not. EFalse is also returned if the 
sl@0
   323
specified path is badly formed. */
sl@0
   324
	{ // static
sl@0
   325
	TParse parse;
sl@0
   326
	TInt retcode;
sl@0
   327
	retcode = parse.Set(aPath, NULL, NULL);
sl@0
   328
	if (retcode != KErrNone)
sl@0
   329
		return EFalse;
sl@0
   330
	if ((! parse.DrivePresent()) || (parse.NameOrExtPresent()))
sl@0
   331
		return EFalse;
sl@0
   332
	if (parse.Path().Length() == 0)
sl@0
   333
		return EFalse;
sl@0
   334
	TFileName dirName = parse.DriveAndPath();
sl@0
   335
	if (dirName.Length() > KMaxFileName)
sl@0
   336
		return(EFalse);
sl@0
   337
    RDir dir;
sl@0
   338
    retcode = dir.Open(aFs,dirName,0);
sl@0
   339
	if (retcode == KErrNone)
sl@0
   340
		dir.Close();
sl@0
   341
	return (retcode == KErrNone);
sl@0
   342
	}
sl@0
   343
sl@0
   344
EXPORT_C void BaflUtils::EnsurePathExistsL(RFs& aFileSession,const TDesC& aFileName)
sl@0
   345
/** Makes one or more directories, if they do not already exist. 
sl@0
   346
	
sl@0
   347
Any valid path component in the specified path that does not already exist 
sl@0
   348
is created as a directory. If the specified path already exists, the function 
sl@0
   349
returns normally.
sl@0
   350
sl@0
   351
@param aFs File server session
sl@0
   352
@param aFileName Path to ensure exists
sl@0
   353
@see RFs::MkDirAll() */
sl@0
   354
	{ // static
sl@0
   355
	TInt error=aFileSession.MkDirAll(aFileName);
sl@0
   356
	if (error!=KErrAlreadyExists)
sl@0
   357
		User::LeaveIfError(error);
sl@0
   358
	}
sl@0
   359
sl@0
   360
EXPORT_C TPtrC BaflUtils::ExtractAppNameFromFullName(const TFullName &aName)
sl@0
   361
/** Gets the application name from a full thread name.
sl@0
   362
sl@0
   363
@param aName Thread name
sl@0
   364
@return Application name
sl@0
   365
@see RThread */
sl@0
   366
	{
sl@0
   367
	// static - return the app name (after first :: before next ::, if any) from a full thread name
sl@0
   368
	TChar delimiter=':';
sl@0
   369
	TInt start=aName.Locate(delimiter);
sl@0
   370
	if (start<0)
sl@0
   371
		start=0; // should never happen
sl@0
   372
	else if (aName.Length()>start+2)
sl@0
   373
		start+=2;
sl@0
   374
	TPtrC rest=aName.Mid(start);
sl@0
   375
	TInt end=rest.Locate(delimiter);
sl@0
   376
	return end<0 ? rest : rest.Left(end);
sl@0
   377
	}
sl@0
   378
sl@0
   379
LOCAL_C TBool IsLanguageExtended(const TLanguage aLanguage)
sl@0
   380
	{
sl@0
   381
	// For compatibility reasons, ELangNone is 0xFFFF. However, it's not an extended language.
sl@0
   382
	if ((aLanguage==ELangNone) || ((static_cast<TUint>(aLanguage))<=KDialectMask))
sl@0
   383
		return EFalse;
sl@0
   384
	else
sl@0
   385
		return ETrue;
sl@0
   386
	}
sl@0
   387
sl@0
   388
sl@0
   389
LOCAL_C TLanguage BaseLanguage(const TLanguage aLanguage)
sl@0
   390
	{
sl@0
   391
	if (IsLanguageExtended(aLanguage))
sl@0
   392
		return static_cast<TLanguage>(aLanguage & KDialectMask);
sl@0
   393
	else
sl@0
   394
		return aLanguage;
sl@0
   395
	}
sl@0
   396
sl@0
   397
LOCAL_C TLanguage NextLanguage(TLanguage aLanguage)
sl@0
   398
/** Returns the next best language to use after aLanguage,
sl@0
   399
based on Symbian's base table of language near-equivalence.
sl@0
   400
@internalAll */
sl@0
   401
	{
sl@0
   402
	switch (aLanguage)
sl@0
   403
		{
sl@0
   404
		case ELangAustralian:
sl@0
   405
		case ELangNewZealand:
sl@0
   406
		case ELangSouthAfricanEnglish:
sl@0
   407
		case ELangInternationalEnglish:
sl@0
   408
		case ELangAmerican:
sl@0
   409
		case ELangEnglish_Apac:
sl@0
   410
		case ELangEnglish_Taiwan:
sl@0
   411
		case ELangEnglish_HongKong:
sl@0
   412
		case ELangEnglish_Prc:
sl@0
   413
		case ELangEnglish_Japan:
sl@0
   414
		case ELangEnglish_Thailand:
sl@0
   415
			return ELangEnglish;
sl@0
   416
		case ELangCanadianEnglish:
sl@0
   417
			return ELangAmerican;	// 2-stage downgrade
sl@0
   418
		case ELangSwissFrench:
sl@0
   419
		case ELangBelgianFrench:
sl@0
   420
		case ELangInternationalFrench:
sl@0
   421
		case ELangCanadianFrench:
sl@0
   422
			return ELangFrench;
sl@0
   423
		case ELangSwissGerman:
sl@0
   424
		case ELangAustrian:
sl@0
   425
			return ELangGerman;
sl@0
   426
		case ELangInternationalSpanish:
sl@0
   427
		case ELangLatinAmericanSpanish:
sl@0
   428
			return ELangSpanish;
sl@0
   429
		case ELangSwissItalian:
sl@0
   430
			return ELangItalian;
sl@0
   431
		case ELangFinlandSwedish:
sl@0
   432
			return ELangSwedish;
sl@0
   433
		case ELangCyprusTurkish:
sl@0
   434
			return ELangTurkish;
sl@0
   435
		case ELangBelgianFlemish:
sl@0
   436
			return ELangDutch;
sl@0
   437
		case ELangHongKongChinese:
sl@0
   438
			return ELangTaiwanChinese;
sl@0
   439
		case ELangCyprusGreek:
sl@0
   440
			return ELangGreek;
sl@0
   441
		case ELangMalay_Apac:
sl@0
   442
			return ELangMalay;
sl@0
   443
		case ELangBrazilianPortuguese:
sl@0
   444
			return ELangPortuguese;
sl@0
   445
		default:
sl@0
   446
			return ELangNone;	
sl@0
   447
		}
sl@0
   448
	}
sl@0
   449
sl@0
   450
sl@0
   451
void AddLanguage(TLanguagePath& aPath, TLanguage aNewLanguage)
sl@0
   452
/** Add language to the language path if there is space.
sl@0
   453
The first empty slot must have "ELangNone" in it. This will also be true
sl@0
   454
on exit. */	
sl@0
   455
	{
sl@0
   456
	TLanguage *p = aPath;
sl@0
   457
	const TLanguage *end = &(aPath[KMaxDowngradeLanguages]);
sl@0
   458
	while (p != end)
sl@0
   459
		{
sl@0
   460
		if (*p == aNewLanguage)
sl@0
   461
			// language already in list
sl@0
   462
			break;
sl@0
   463
		if (*p == ELangNone)
sl@0
   464
			{
sl@0
   465
			// found the end of the list
sl@0
   466
			p[0] = aNewLanguage;
sl@0
   467
			p[1] = ELangNone;
sl@0
   468
			break;
sl@0
   469
			}
sl@0
   470
		++p;
sl@0
   471
		}
sl@0
   472
	return;
sl@0
   473
	}
sl@0
   474
sl@0
   475
void MakeLanguageDowngradePath(TLanguagePath& aPath,
sl@0
   476
	TLanguage aCurrent, TLanguage aIdeal, const TLocale& aLocale)
sl@0
   477
	{
sl@0
   478
	TInt j = 0;
sl@0
   479
	if( aIdeal != ELangNone)
sl@0
   480
		{
sl@0
   481
		aPath[j++]=aIdeal;	
sl@0
   482
		}
sl@0
   483
	aPath[j++] = aCurrent;
sl@0
   484
	aPath[j++] = ELangNone;
sl@0
   485
sl@0
   486
	if (aCurrent & ~KDialectMask)
sl@0
   487
		AddLanguage(aPath, static_cast<TLanguage>(aCurrent & KDialectMask));
sl@0
   488
sl@0
   489
	for (TInt i=0;i<=2;i++)
sl@0
   490
		{
sl@0
   491
		AddLanguage(aPath, aLocale.LanguageDowngrade(i));
sl@0
   492
		AddLanguage(aPath, BaseLanguage(aLocale.LanguageDowngrade(i)));
sl@0
   493
		}
sl@0
   494
sl@0
   495
	while (ELangNone != (aCurrent = NextLanguage(BaseLanguage(aCurrent))))  
sl@0
   496
		AddLanguage(aPath, aCurrent);
sl@0
   497
	}
sl@0
   498
sl@0
   499
TInt RRealDirectoryScanner::Open(RFs& aFs, const TDesC& aMatchPattern)
sl@0
   500
	{
sl@0
   501
	return iDir.Open(aFs, aMatchPattern,
sl@0
   502
		KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive);
sl@0
   503
	}
sl@0
   504
sl@0
   505
TInt RRealDirectoryScanner::Next(TEntry& aOut)
sl@0
   506
	{
sl@0
   507
	return iDir.Read(aOut);
sl@0
   508
	}
sl@0
   509
sl@0
   510
void RRealDirectoryScanner::Close()
sl@0
   511
	{
sl@0
   512
	iDir.Close();
sl@0
   513
	}
sl@0
   514
sl@0
   515
/**
sl@0
   516
Simply counts the number of numerical characters at the end of the name passed.
sl@0
   517
sl@0
   518
@internalComponent
sl@0
   519
@param			aFilename The filename to parse
sl@0
   520
				
sl@0
   521
@return			Count of the numeric digits at the end of the name passed, 
sl@0
   522
				e.g. x.r491 gives 3.
sl@0
   523
*/
sl@0
   524
TInt TNearestLanguageFileFinder::CountDigitsFromEnd(const TDesC& aFilename)
sl@0
   525
	{
sl@0
   526
	TInt digitCount = 0;
sl@0
   527
	
sl@0
   528
	for (TInt idx = aFilename.Length()-1; idx>=0 && ISDIGIT (aFilename [idx]); --idx)
sl@0
   529
		{
sl@0
   530
		++digitCount;
sl@0
   531
		}
sl@0
   532
		
sl@0
   533
	return digitCount;
sl@0
   534
	}
sl@0
   535
sl@0
   536
sl@0
   537
/**
sl@0
   538
Counts the number of digits at the end of a filename.
sl@0
   539
sl@0
   540
@internalComponent
sl@0
   541
@param			aFilename The filename to parse
sl@0
   542
				
sl@0
   543
@return			Count of the numeric digits at the end of the suffix, 
sl@0
   544
				e.g. x.r491 gives 3.
sl@0
   545
				0 if no numeric end of suffix,
sl@0
   546
				KErrBadName for an invalid filename,
sl@0
   547
				KErrNotSupported if the filename (minus path) is less
sl@0
   548
				than or equal to KInvNameAndMinSuffixLength in length
sl@0
   549
*/
sl@0
   550
TInt TNearestLanguageFileFinder::CountDigitsFromEndInSuffix (const TDesC& aFilename)
sl@0
   551
	{
sl@0
   552
	TInt digitCount = 0;
sl@0
   553
	TInt slashIdx = 0;
sl@0
   554
	TInt len = aFilename.Length ();
sl@0
   555
	
sl@0
   556
	// NOTE: We didn't use TChar here as they are too slow.
sl@0
   557
	// We also didn't use TParse as they are too large.
sl@0
   558
	
sl@0
   559
	// don't work on the path
sl@0
   560
	for (slashIdx=len-1; slashIdx >= 0 && aFilename[slashIdx] != '\\'; --slashIdx)
sl@0
   561
	{/*do nothing*/};
sl@0
   562
	
sl@0
   563
	// Get new length
sl@0
   564
	if (slashIdx>=0) {len = len-slashIdx-1;}
sl@0
   565
sl@0
   566
	// Initial test to see if filename legal size.
sl@0
   567
	if (len > KInvNameAndMinSuffixLength)
sl@0
   568
		{
sl@0
   569
		digitCount = CountDigitsFromEnd(aFilename);
sl@0
   570
sl@0
   571
		// Can't store something bigger or we'll panic!
sl@0
   572
		if (digitCount > KMaxSuffixLength)
sl@0
   573
			{
sl@0
   574
			digitCount = KErrBadName;
sl@0
   575
			}
sl@0
   576
		else
sl@0
   577
		// numeric filename, e.g. "1234". 
sl@0
   578
		// No preceeding alpha character
sl@0
   579
		if (!(len-digitCount))
sl@0
   580
			{
sl@0
   581
			digitCount = KErrBadName;
sl@0
   582
			}
sl@0
   583
		}
sl@0
   584
	else
sl@0
   585
		{
sl@0
   586
		digitCount = KErrNotSupported;
sl@0
   587
		}
sl@0
   588
		
sl@0
   589
	return digitCount;
sl@0
   590
	}
sl@0
   591
sl@0
   592
RDirectoryScanner& TNearestLanguageFileFinder::DirectoryScanner()
sl@0
   593
	{
sl@0
   594
	return iDirScanner;
sl@0
   595
	}
sl@0
   596
sl@0
   597
TBool TNearestLanguageFileFinder::FileExists(const TDesC& aFileName) const
sl@0
   598
	{
sl@0
   599
	return BaflUtils::FileExists(iFs, aFileName);
sl@0
   600
	}
sl@0
   601
sl@0
   602
TBool TNearestLanguageFileFinder::FindDrive()
sl@0
   603
	{
sl@0
   604
	ASSERT(iFileName);
sl@0
   605
	TBool found=EFalse;
sl@0
   606
	TInt driveLength=iDrives.Length();
sl@0
   607
	for (TInt drive = 0; drive!=driveLength; ++drive)
sl@0
   608
		{
sl@0
   609
		(*iFileName)[0] = iDrives[drive];
sl@0
   610
		if (FileExists(*iFileName))
sl@0
   611
			{
sl@0
   612
			found=ETrue;
sl@0
   613
			break;
sl@0
   614
			}
sl@0
   615
		}
sl@0
   616
	return found;
sl@0
   617
	}
sl@0
   618
sl@0
   619
TBool TNearestLanguageFileFinder::AppendLanguageCode(TLanguage aLanguage)
sl@0
   620
	{
sl@0
   621
	TInt rest = static_cast<TInt>(aLanguage);
sl@0
   622
#ifdef _DEBUG
sl@0
   623
	_LIT(KErrorMessage, "Bafl");
sl@0
   624
#endif
sl@0
   625
	__ASSERT_DEBUG(0 <= rest, User::Panic(KErrorMessage,KErrArgument));
sl@0
   626
	iFileName->SetLength(iBaseLength);
sl@0
   627
	const TInt remaining = iFileName->MaxLength() - iBaseLength;
sl@0
   628
	TInt soFar = 0;
sl@0
   629
	TBuf<1> num;
sl@0
   630
	num.Append('0');
sl@0
   631
	TBool appendLangSuccess = ETrue;
sl@0
   632
	TInt digitCount = 0;
sl@0
   633
	TInt digit = 0;
sl@0
   634
	while (rest)
sl@0
   635
		{
sl@0
   636
		if (remaining == soFar)
sl@0
   637
			{
sl@0
   638
			// no more room in descriptor- return rather than panic,
sl@0
   639
			// file cannot exist.
sl@0
   640
			iFileName->SetLength(iBaseLength);
sl@0
   641
			appendLangSuccess= EFalse;
sl@0
   642
			break;
sl@0
   643
			}
sl@0
   644
		// Convert the number to ASCII by consistantly getting the base 10 remainder to convert.
sl@0
   645
		// The number is updated minus the remainder for the next iteration.
sl@0
   646
		// eg (rest = 123) -> (12, r3) -> (1, r2) -> (0, r1)
sl@0
   647
		// Then insert the ASCII representation of the remainder into the filename end
sl@0
   648
		// so it appears the correct way round.
sl@0
   649
		// eg (filename.r) -> (filename.r3) -> (filename.r23) -> (filename.r123)
sl@0
   650
		digit = rest % 10;
sl@0
   651
		digitCount++;
sl@0
   652
		rest /= 10;
sl@0
   653
		num[0] = static_cast<TText16>(digit + '0');
sl@0
   654
		iFileName->Insert(iBaseLength, num);
sl@0
   655
sl@0
   656
		// Minimum suffix length is KInvNameAndMinSuffixLength
sl@0
   657
		// so we have to insert zeros to make this up.
sl@0
   658
		while (!rest && digitCount < KInvNameAndMinSuffixLength)
sl@0
   659
			{
sl@0
   660
			num[0] = static_cast<TText16>('0');
sl@0
   661
			iFileName->Insert(iBaseLength, num);
sl@0
   662
			++digitCount;
sl@0
   663
			}
sl@0
   664
			
sl@0
   665
		++soFar;
sl@0
   666
		}
sl@0
   667
		
sl@0
   668
	return appendLangSuccess;
sl@0
   669
	}
sl@0
   670
sl@0
   671
sl@0
   672
TBool TNearestLanguageFileFinder::FindLanguageAndDrive()
sl@0
   673
/** Search for files across all drives in all languages in the path plus the
sl@0
   674
language-neutral file. */
sl@0
   675
	{
sl@0
   676
	ASSERT(iFileName);
sl@0
   677
	// No point appending if the suffix is bad
sl@0
   678
	for (const TLanguage* currentLang = iPath; *currentLang != ELangNone; ++currentLang)
sl@0
   679
		{
sl@0
   680
		if (AppendLanguageCode(*currentLang) && FindDrive())
sl@0
   681
			{
sl@0
   682
			iLanguage = *currentLang;
sl@0
   683
			return ETrue;
sl@0
   684
			}
sl@0
   685
		}
sl@0
   686
	// search for language-neutral file
sl@0
   687
	iFileName->SetLength(iBaseLength);
sl@0
   688
	iFileName->Append(iSuffix);
sl@0
   689
	return FindDrive();
sl@0
   690
	}
sl@0
   691
sl@0
   692
TInt TNearestLanguageFileFinder::LanguageNumberFromFile(const TDesC& aFileName, const TDesC& aStem)
sl@0
   693
	{
sl@0
   694
	TInt lang = 0;
sl@0
   695
	TInt multiplier = 1;
sl@0
   696
	TInt leadingZeroCount = 0;
sl@0
   697
	TInt languageNumber = KErrNotFound;
sl@0
   698
	const TText* firstChar = aFileName.Ptr();
sl@0
   699
	const TText* lastChar = firstChar + aFileName.Length() - 1;
sl@0
   700
	const TText* currentChar = lastChar;
sl@0
   701
	// string cannot contain only numbers, because it must have a ':' in it
sl@0
   702
	while ('0' <= *currentChar && *currentChar <= '9')
sl@0
   703
		{
sl@0
   704
		if (*currentChar == '0')
sl@0
   705
			leadingZeroCount++;
sl@0
   706
		else
sl@0
   707
			{
sl@0
   708
			leadingZeroCount = 0;
sl@0
   709
			lang += multiplier * (*currentChar - '0');
sl@0
   710
			}
sl@0
   711
		multiplier *= 10;
sl@0
   712
		--currentChar;
sl@0
   713
		}
sl@0
   714
	TInt along=lastChar - currentChar;
sl@0
   715
	if (2 <= along)
sl@0
   716
		{
sl@0
   717
		// We have at least 2 digits at the end.
sl@0
   718
		// trim of bad leading zeros
sl@0
   719
		TInt maxTrim = along - 2;
sl@0
   720
		if (maxTrim < leadingZeroCount)
sl@0
   721
			{
sl@0
   722
			leadingZeroCount = maxTrim;
sl@0
   723
			}
sl@0
   724
		currentChar += leadingZeroCount;
sl@0
   725
		// we have at least 2 digits at the end but does the rest of it match the stem?
sl@0
   726
		TPtrC foundStem(firstChar, currentChar - firstChar + 1);
sl@0
   727
		//foundStem.CompareF(aStem.Right(foundStem.Length()))
sl@0
   728
		if (0 == foundStem.CompareF(aStem))
sl@0
   729
			{
sl@0
   730
			languageNumber=lang;
sl@0
   731
			}
sl@0
   732
		}
sl@0
   733
	return languageNumber;
sl@0
   734
	}
sl@0
   735
sl@0
   736
TInt TNearestLanguageFileFinder::FindFirstLanguageFile(RFs& aFs)
sl@0
   737
	{
sl@0
   738
	ASSERT(iFileName);
sl@0
   739
	iFileName->SetLength(iBaseLength);
sl@0
   740
	TPtrC name(*iFileName);
sl@0
   741
	TParsePtrC nameToParse(name);
sl@0
   742
	TPtrC nameStem(nameToParse.NameAndExt());
sl@0
   743
	iFileName->Append('*');
sl@0
   744
	TInt bestLanguageMatch = KMaxTInt;
sl@0
   745
	RDirectoryScanner& scanner = DirectoryScanner();
sl@0
   746
	TInt err = scanner.Open(aFs, *iFileName);
sl@0
   747
	if (err != KErrNone)
sl@0
   748
		{
sl@0
   749
		return err;
sl@0
   750
		}
sl@0
   751
	TEntry entry;
sl@0
   752
	while (KErrNone == scanner.Next(entry))
sl@0
   753
		{
sl@0
   754
		TInt lang = LanguageNumberFromFile(entry.iName, nameStem);
sl@0
   755
		if (0 < lang && lang < bestLanguageMatch)
sl@0
   756
			{
sl@0
   757
			bestLanguageMatch = lang;
sl@0
   758
			}
sl@0
   759
		}
sl@0
   760
	scanner.Close();
sl@0
   761
	if (bestLanguageMatch != KMaxTInt)
sl@0
   762
		{
sl@0
   763
		iLanguage = static_cast<TLanguage>(bestLanguageMatch);
sl@0
   764
		AppendLanguageCode(static_cast<TLanguage>(bestLanguageMatch));
sl@0
   765
		return KErrNone;
sl@0
   766
		}
sl@0
   767
	return KErrNotFound;
sl@0
   768
	}
sl@0
   769
sl@0
   770
// Try each drive for any language files
sl@0
   771
// iFileName must have a directory specifier
sl@0
   772
TInt TNearestLanguageFileFinder::FindFirstLanguageFileAndDrive()
sl@0
   773
	{
sl@0
   774
	ASSERT(iFileName);
sl@0
   775
	TInt findFirstResult=KErrNotFound;
sl@0
   776
	TInt driveLength=iDrives.Length();
sl@0
   777
	for (TInt drive = 0; drive != driveLength; ++drive)
sl@0
   778
		{
sl@0
   779
		(*iFileName)[0] = iDrives[drive];
sl@0
   780
		TInt err = FindFirstLanguageFile(CONST_CAST(RFs&,iFs));
sl@0
   781
		if (err == KErrNone || err == KErrNoMemory)
sl@0
   782
			{
sl@0
   783
			findFirstResult=err;
sl@0
   784
			break;
sl@0
   785
			}
sl@0
   786
		}
sl@0
   787
	return findFirstResult;
sl@0
   788
	}
sl@0
   789
sl@0
   790
/**
sl@0
   791
Invalid filenames are any filename whose length (minus path) must be greater
sl@0
   792
than KInvNameAndMinSuffixLength, and whose form is purely numerical, i.e. '1234' 
sl@0
   793
*/
sl@0
   794
TBool TNearestLanguageFileFinder::SetFileName(TFileName& aFileName)
sl@0
   795
	{
sl@0
   796
	iDrives.Zero();
sl@0
   797
	iFileName = &aFileName;
sl@0
   798
	iOriginalBaseLength = iFileName->Length();
sl@0
   799
	
sl@0
   800
	TInt suffixLength = CountDigitsFromEndInSuffix (aFileName);
sl@0
   801
	
sl@0
   802
	// No point trying for filenames thats are badly formed
sl@0
   803
	// or that are too large.
sl@0
   804
	if (suffixLength >= 0 && 
sl@0
   805
		KInvNameAndMinSuffixLength < iOriginalBaseLength)
sl@0
   806
		{
sl@0
   807
		if (suffixLength > 0)
sl@0
   808
			{
sl@0
   809
			// all of suffix to be replaced 
sl@0
   810
			iSuffix = iFileName->Right(suffixLength);
sl@0
   811
			iOriginalBaseLength -= suffixLength;
sl@0
   812
			iFileName->SetLength(iOriginalBaseLength);
sl@0
   813
			}
sl@0
   814
		else
sl@0
   815
			{ 
sl@0
   816
			// No numerical part to suffix
sl@0
   817
			TInt periodIdx = 0;
sl@0
   818
			
sl@0
   819
			// Search for the period within range KInvNameAndMinSuffixLength 
sl@0
   820
			// from the end. As this must work for all values of
sl@0
   821
			// KInvNameAndMinSuffixLength
sl@0
   822
			for (TInt i = iOriginalBaseLength-1; 
sl@0
   823
			     !periodIdx && i >= (iOriginalBaseLength-KInvNameAndMinSuffixLength-1);
sl@0
   824
			     --i)
sl@0
   825
				{
sl@0
   826
				if ((*iFileName) [i] == '.')
sl@0
   827
					{
sl@0
   828
					periodIdx = i;
sl@0
   829
					}
sl@0
   830
				}
sl@0
   831
			
sl@0
   832
			// Don't handle files ending in a period.
sl@0
   833
			// This is because the behaviour is different between Windows
sl@0
   834
			// and Symbian Fs. In Windows it strips the period off.
sl@0
   835
			//
sl@0
   836
			// However, and this shouldn't happen as it is not shown
sl@0
   837
			// (in the documentation) to be valid.
sl@0
   838
			// Just try our best.
sl@0
   839
			if (periodIdx == iOriginalBaseLength-1)
sl@0
   840
				{
sl@0
   841
				iSuffix.Zero();
sl@0
   842
				return EFalse;
sl@0
   843
				}
sl@0
   844
			else
sl@0
   845
			if (periodIdx)
sl@0
   846
				{
sl@0
   847
				// If there are KInvNameAndMinSuffixLength chars after the period
sl@0
   848
				// simply replace them.
sl@0
   849
				TInt right = iOriginalBaseLength-periodIdx-1;
sl@0
   850
				iSuffix = iFileName->Right(right);
sl@0
   851
				iOriginalBaseLength -= right;
sl@0
   852
				iFileName->SetLength(iOriginalBaseLength);					
sl@0
   853
				}
sl@0
   854
			else
sl@0
   855
				{
sl@0
   856
				// Make the suffix start from KInvNameAndMinSuffixLength 
sl@0
   857
				// from the right
sl@0
   858
				TInt right = KInvNameAndMinSuffixLength;
sl@0
   859
				iSuffix = iFileName->Right(right);
sl@0
   860
				iOriginalBaseLength -= right;
sl@0
   861
				iFileName->SetLength(iOriginalBaseLength);					
sl@0
   862
				}
sl@0
   863
			}
sl@0
   864
		}
sl@0
   865
	else
sl@0
   866
		{
sl@0
   867
		// bad or no suffix - treat the same
sl@0
   868
		iSuffix.Zero();
sl@0
   869
		return EFalse;
sl@0
   870
		}
sl@0
   871
sl@0
   872
	// For filenames with no drive letter prefix and also for filenames
sl@0
   873
	// shorter than the drive letter length, i.e. with no drive
sl@0
   874
	// information, insert it.
sl@0
   875
	// Handles if the user simply enters the drive, e.g. "c:".
sl@0
   876
	if (iOriginalBaseLength < KMaxDriveName || (*iFileName)[1] != ':')
sl@0
   877
		{
sl@0
   878
		// Set up the default if none supplied and make room in the filename 
sl@0
   879
        // array to contain a drive specification. Set initial drive letter to -1
sl@0
   880
        // so the iFileName is repaired before exited 
sl@0
   881
		iInitialDriveLetter = -1;
sl@0
   882
		iFileName->Insert(0, _L("_:")); 
sl@0
   883
		iDrives.Append('Z');
sl@0
   884
		}
sl@0
   885
	else
sl@0
   886
		{
sl@0
   887
	   // Use the drive supplied inthe aName to NearestLanguageFile()
sl@0
   888
		iInitialDriveLetter = (*iFileName)[0];
sl@0
   889
		iDrives.Append(iInitialDriveLetter);
sl@0
   890
		}
sl@0
   891
	
sl@0
   892
	iBaseLength = iFileName->Length();
sl@0
   893
	
sl@0
   894
	return ETrue;
sl@0
   895
	}
sl@0
   896
sl@0
   897
sl@0
   898
TLanguage TNearestLanguageFileFinder::Language()
sl@0
   899
	{
sl@0
   900
	return iLanguage;
sl@0
   901
	}
sl@0
   902
sl@0
   903
TNearestLanguageFileFinder::TNearestLanguageFileFinder(
sl@0
   904
	const RFs& aFs)
sl@0
   905
	: iFs(aFs), iFileName(0), iLanguage(ELangNone)
sl@0
   906
	{
sl@0
   907
	}
sl@0
   908
sl@0
   909
void TNearestLanguageFileFinder::RepairFileName()
sl@0
   910
	{
sl@0
   911
	ASSERT(iFileName);
sl@0
   912
	iFileName->SetLength(iBaseLength);
sl@0
   913
	if (iInitialDriveLetter == -1)
sl@0
   914
		iFileName->Delete(0, 2);
sl@0
   915
	else
sl@0
   916
		(*iFileName)[0] = static_cast<TText>(iInitialDriveLetter);
sl@0
   917
	iFileName->SetLength(iOriginalBaseLength);
sl@0
   918
	iFileName->Append(iSuffix);
sl@0
   919
	}
sl@0
   920
sl@0
   921
sl@0
   922
/**
sl@0
   923
Add the custom resource drive to the start of the iDrives string.
sl@0
   924
sl@0
   925
The custom resource drive is a preset writeable drive on which customised 
sl@0
   926
resource files may be present. This drive takes priority over the other 
sl@0
   927
drives when searching for language files.
sl@0
   928
sl@0
   929
@return KErrNone if iDrives string was successfully modified; KErrAlreadyExists 
sl@0
   930
if the drive is already present in the string; otherwise one of 
sl@0
   931
the other system-wide error codes (iDrives will be unmodified). 
sl@0
   932
*/
sl@0
   933
TInt TNearestLanguageFileFinder::AddCustomResourceDrive()
sl@0
   934
	{
sl@0
   935
	TInt drive = GetCustomResourceDriveNumber();
sl@0
   936
	if (drive<0)
sl@0
   937
		return drive;
sl@0
   938
	
sl@0
   939
	// if drive not already in drive list
sl@0
   940
	if (iDrives.LocateF('A' + drive) < 0)
sl@0
   941
		{
sl@0
   942
		// add it
sl@0
   943
		_LIT(KDrivePlaceholder, "_");
sl@0
   944
		iDrives.Insert(0, KDrivePlaceholder);
sl@0
   945
		iDrives[0] = 'A' + drive;
sl@0
   946
		return KErrNone;
sl@0
   947
		}
sl@0
   948
	else
sl@0
   949
		return KErrAlreadyExists;
sl@0
   950
	}
sl@0
   951
sl@0
   952
sl@0
   953
void TNearestLanguageFileFinder::AddAllDrives()
sl@0
   954
	{
sl@0
   955
	ASSERT(iDrives.Length() < 2);
sl@0
   956
	if (iDrives.Length() == 0)
sl@0
   957
		{
sl@0
   958
		iDrives = KAllDrives;
sl@0
   959
		return;
sl@0
   960
		}
sl@0
   961
	TInt pos = KAllDrives().LocateF(iDrives[0]);
sl@0
   962
	if (pos < 0)
sl@0
   963
		{
sl@0
   964
		iDrives = KAllDrives;
sl@0
   965
		return;
sl@0
   966
		}
sl@0
   967
	iDrives.Append(KAllDrives().Left(pos));
sl@0
   968
	iDrives.Append(KAllDrives().Mid(pos + 1));
sl@0
   969
	}
sl@0
   970
sl@0
   971
sl@0
   972
/**
sl@0
   973
Get the value of the custom resource drive.
sl@0
   974
sl@0
   975
The custom resource drive is a preset writeable drive on which customised language resource 
sl@0
   976
files can reside. The drive number is accessed via the HAL attribute ECustomResourceDrive. 
sl@0
   977
It is then returned if it has been defined as a valid drive no.
sl@0
   978
Otherwise for backward compatibility reasons an attempt is then made to access the system 
sl@0
   979
drive HAL attribute instead. This drive number is returned if it has been defined as a valid 
sl@0
   980
drive number.  
sl@0
   981
Otherwise if neither a valid ECustomResourceDrive or ESystemDrive exists then KErrNotFound 
sl@0
   982
is returned.
sl@0
   983
 
sl@0
   984
Note that the ESystemDrive HAL attribute has been deprecated. It is accessed here to cater 
sl@0
   985
for existing implementations which still expect it to be used.
sl@0
   986
 
sl@0
   987
@return The drive number (corresponding to a TDriveNumber value) if successful; 
sl@0
   988
KErrNotFound if neither a valid ECustomResourceDrive or a valid ESystemDrive HAL attribute 
sl@0
   989
is defined;  
sl@0
   990
 
sl@0
   991
@see HAL::ECustomResourceDrive
sl@0
   992
@see HAL::ESystemDrive
sl@0
   993
*/
sl@0
   994
TInt TNearestLanguageFileFinder::GetCustomResourceDriveNumber() const
sl@0
   995
	{
sl@0
   996
	TInt drive = KErrNotFound;
sl@0
   997
	
sl@0
   998
	// access custom resource drive attribute  
sl@0
   999
	if (HAL::Get(HAL::ECustomResourceDrive, drive) == KErrNone)
sl@0
  1000
		{
sl@0
  1001
		// check that drive is valid
sl@0
  1002
		if (drive>=EDriveA && drive<=EDriveZ)
sl@0
  1003
			return drive;	 
sl@0
  1004
		}
sl@0
  1005
		    		    
sl@0
  1006
	// access system drive attribute  
sl@0
  1007
	// (Note that ESystemDrive is deprecated. It is checked here 
sl@0
  1008
	// solely for backward compatibility reasons.)		
sl@0
  1009
	if (HAL::Get(HAL::ESystemDrive, drive) == KErrNone)
sl@0
  1010
		{
sl@0
  1011
		// check that drive is valid
sl@0
  1012
		if (drive>=EDriveA && drive<=EDriveZ)
sl@0
  1013
				return drive;
sl@0
  1014
		}		
sl@0
  1015
 
sl@0
  1016
	return KErrNotFound;
sl@0
  1017
	}
sl@0
  1018
sl@0
  1019
sl@0
  1020
sl@0
  1021
/** Get the value of the system drive.
sl@0
  1022
sl@0
  1023
The system drive can be set to one of the built-in read/write drives. Which 
sl@0
  1024
drive is used is hardware-dependent. On some hardware, there may not be a 
sl@0
  1025
system drive. The system drive is used as the drive on which localisable files 
sl@0
  1026
are searched for. This enables a phone to be localised dynamically, using 
sl@0
  1027
files not in the ROM.
sl@0
  1028
sl@0
  1029
@param aDriveNumber On return, contains the drive number of the system drive.
sl@0
  1030
@return KErrNone is always returned.
sl@0
  1031
sl@0
  1032
@deprecated This method has been replaced by (and now internally calls) 
sl@0
  1033
RFs:GetSystemDrive, which always returns a valid drive number.
sl@0
  1034
sl@0
  1035
@see BaflUtils::NearestLanguageFile
sl@0
  1036
@see RFs::GetSystemDrive
sl@0
  1037
*/
sl@0
  1038
EXPORT_C TInt BaflUtils::GetSystemDrive(TDriveNumber& aDriveNumber)
sl@0
  1039
	{
sl@0
  1040
	aDriveNumber = RFs::GetSystemDrive();
sl@0
  1041
	return KErrNone;
sl@0
  1042
	}
sl@0
  1043
sl@0
  1044
sl@0
  1045
/** Set most appropriate extension language code for filename and set corresponding language.
sl@0
  1046
sl@0
  1047
Symbian uses numeric values to identify natural languages as specified by the TLanguage enumeration
sl@0
  1048
defined in e32const.h. These values are used at the end of filename extensions to identify the
sl@0
  1049
languages pertaining to files which have language specific variants such as resource files.
sl@0
  1050
For instance filename.r01 and filename.r02 would be the English and French versions of the
sl@0
  1051
resource file filename.rsc. Language codes can be between 2 to 5 digits in length.
sl@0
  1052
sl@0
  1053
Starting from Symbian OS v7.0 this function constructs and uses a language downgrade path which 
sl@0
  1054
consists of up to sixteen TLanguage values the first of which is the ideal language followed by 
sl@0
  1055
the language of the current locale. Up to the next three can be customised using 
sl@0
  1056
TLocale::SetLanguageDowngrade(). The rest of the language downgrade path is based on a 
sl@0
  1057
table of language near equivalence which is internal to Symbian.
sl@0
  1058
sl@0
  1059
This function searches the custom resource drive (if set, retrieved from HAL)
sl@0
  1060
and then searches the optional drive specified in aName or 'Z:' if none is 
sl@0
  1061
specified in aName. The custom resource drive is retrieved from the HAL 
sl@0
  1062
attribute ECustomResourceDrive if set, if not set it will retrieve the legacy 
sl@0
  1063
value set in the legacy HAL attribute ESystemDrive. No custom resource drive 
sl@0
  1064
is searched if neither are set.  
sl@0
  1065
Note - setting the custom resource drive will reduce the performance of this 
sl@0
  1066
routine which will adversely affect device performance e.g. at boot up.
sl@0
  1067
On NAND Flash based devices use of a composite Z: drive file system made up of multiple 
sl@0
  1068
ROM images is the preferred mechanism for customising language resources on devices in 
sl@0
  1069
Symbian OS 9.2 onwards, see Developer Library » Base Porting Guide » Porting: background 
sl@0
  1070
information » NAND flash » NAND Flash image format. Thus use of the custom resource drive 
sl@0
  1071
HAL attribute is effectively obsolete.
sl@0
  1072
  
sl@0
  1073
The last two characters of aName are removed along with any digits which appear before them.
sl@0
  1074
Then language codes specified in the constructed language downgrade path are appended in turn to 
sl@0
  1075
aName as a match is searched for in the file system. In case no match is found using the 
sl@0
  1076
constructed language downgradepath then files in the specified directory are searched for a 
sl@0
  1077
suitable extension with preference given to the one specified if present. In cases where a 
sl@0
  1078
match takes place the aName and aLanguage arguments are updated otherwise aName is left 
sl@0
  1079
unchanged and aLanguage is set to ELangNone.
sl@0
  1080
sl@0
  1081
Here are some examples of correct and incorrect function usage with different aName inputs, 
sl@0
  1082
file system state and downgrade paths as follows: 
sl@0
  1083
sl@0
  1084
@code
sl@0
  1085
Following files exist:
sl@0
  1086
C:\\abc.rsc  - Language Neutral resource file.
sl@0
  1087
C:\\abc.r01  - Resource file for the English language.
sl@0
  1088
C:\\abc.r10  - Resource file for the American-English language.
sl@0
  1089
C:\\abc.r160 - Resource file for the English as appropriate in Japan.
sl@0
  1090
sl@0
  1091
Constructed Downgrade Language Path cases:
sl@0
  1092
- Case 1. (ELangAmerican -> ELangEnglish -> ELangNone).
sl@0
  1093
- Case 2. (ELangEnglish_Japan -> ELangEnglish -> ELangNone).
sl@0
  1094
- Case 3. Same as case 1, However "C:\\abc.r10" is deleted prior to the function call.
sl@0
  1095
- Case 4. Same as case 1, However both "C:\\abc.r01" and "C:\\abc.r10" are deleted prior to the function call.
sl@0
  1096
@endcode
sl@0
  1097
sl@0
  1098
@code
sl@0
  1099
Input aName . . . . Output aName. . . aLanguage . . . . . Description
sl@0
  1100
--------------------------------------------------------------------------------------------------------------------
sl@0
  1101
"C:\\abc.rsc" . . . "C:\\abc.r10" . . ELangAmerican . . . Match on first language (Case 1)
sl@0
  1102
"C:\\abc.r10" . . . "C:\\abc.r10" . . ELangAmerican . . . Match, However it's not the intended use of 
sl@0
  1103
. . . . . . . . . . . . . . . . . . . . . . . . . . . . . the function (Case 1)
sl@0
  1104
"C:\\abc.r" . . . . "C:\\abc.r" . . . ELangNone . . . . . The no. of characters in the suffix is less than 
sl@0
  1105
. . . . . . . . . . . . . . . . . . . . . . . . . . . . . KInvNameAndMinSuffixLength(2)(Case 1)
sl@0
  1106
"C:\\abc.". . . . . "C:\\abc.". . . . ELangNone . . . . . Invalid Suffix: The filename ends with a period(Case 1)
sl@0
  1107
"C:\\abc.r123456" . "C:\\abc.r123456" ELangNone . . . . . Invalid Suffix: The no. of digits in the suffix is greater 
sl@0
  1108
. . . . . . . . . . . . . . . . . . . . . . . . . . . . . than KMaxSuffixLength(5) (Case 1)
sl@0
  1109
"C:\\abc.10". . . . "C:\\abc.10 . . . ELangNone . . . . . Invalid Suffix: There's no proceeding alphabetical 
sl@0
  1110
. . . . . . . . . . . . . . . . . . . . . . . . . . . . . characters in the suffix (Case 1)
sl@0
  1111
"\\abc.rsc" . . . . "\\abc.rsc" . . . ELangNone . . . . . No drive so Z: search, no match (Case 1)
sl@0
  1112
"C:\\abc.rsc" . . . "C:\\abc.r160". . ELangEnglish_Japan. Match for language file 3 digits long (Case 2)
sl@0
  1113
"C:\\abc.rsc" . . . "C:\\abc.r01" . . ELangEnglish. . . . Match on second language (Case 3)
sl@0
  1114
"C:\\abc.rsc" . . . "C:\\abc.rsc" . . ELangNone . . . . . No corresponding langauge file match found (Case 4)
sl@0
  1115
---------------------------------------------------------------------------------------------------------------------
sl@0
  1116
@endcode
sl@0
  1117
sl@0
  1118
@param aFs File server session.
sl@0
  1119
@param aName Optional drive specification, followed by optional path name,
sl@0
  1120
followed by basename for filename, followed by period and extension.
sl@0
  1121
On return, in case of a match, this is replaced by the language-specific version
sl@0
  1122
which consists of the last two characters of the extension plus any preceding
sl@0
  1123
numeric characters being replaced by the language code. Remains unchanged when there's no match 
sl@0
  1124
@param aLanguage On return, in case of a match, this is replaced by the corresponding language.
sl@0
  1125
In case of no match it's set to ELangNone.
sl@0
  1126
@see TLanguage
sl@0
  1127
@see BaflUtils::GetDowngradePathL
sl@0
  1128
@see TLocale::SetLanguageDowngrade
sl@0
  1129
*/
sl@0
  1130
EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage)
sl@0
  1131
	{
sl@0
  1132
#if defined(DO_PROFILING)
sl@0
  1133
	RDebug::ProfileReset(FIRST_PROFILE_INDEX, PROFILE_COUNT);
sl@0
  1134
	RDebug::ProfileStart(PROFILE_INDEX_1);
sl@0
  1135
#endif
sl@0
  1136
	TNearestLanguageFileFinder finder(aFs);
sl@0
  1137
	TBool goodSuffix=finder.SetFileName(aName);
sl@0
  1138
	
sl@0
  1139
	// Only continue if the suffix is good.
sl@0
  1140
	if(goodSuffix)
sl@0
  1141
		{
sl@0
  1142
		// add preset customised resource drive to drive list  
sl@0
  1143
		// Note that errors returned from AddCustomResourceDrive are ignored. This is because if 
sl@0
  1144
		// a custom resource drive has not been found we still want to continue on with searching 
sl@0
  1145
		// other drives according to our algorithm
sl@0
  1146
		finder.AddCustomResourceDrive();
sl@0
  1147
		
sl@0
  1148
		TLocale locale;
sl@0
  1149
		TLanguage idealLanguage;
sl@0
  1150
		idealLanguage = IdealLanguage();
sl@0
  1151
		MakeLanguageDowngradePath(finder.iPath, User::Language(), idealLanguage, locale);
sl@0
  1152
		if (!finder.FindLanguageAndDrive()
sl@0
  1153
			&& KErrNone != finder.FindFirstLanguageFileAndDrive())
sl@0
  1154
			finder.RepairFileName();
sl@0
  1155
		aLanguage = finder.Language();
sl@0
  1156
		}
sl@0
  1157
		
sl@0
  1158
#if defined(DO_PROFILING)
sl@0
  1159
	RDebug::ProfileEnd(PROFILE_INDEX_1);
sl@0
  1160
	TProfile profile[PROFILE_COUNT];
sl@0
  1161
	RDebug::ProfileResult(&profile[0], FIRST_PROFILE_INDEX, PROFILE_COUNT);
sl@0
  1162
	if(goodSuffix)
sl@0
  1163
		{
sl@0
  1164
		RDebug::Print(_L("BaflUtils::NearestLanguageFile profile info: %d.%03ds"), profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime/1000000, profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime%1000000);
sl@0
  1165
		}
sl@0
  1166
	else
sl@0
  1167
		{
sl@0
  1168
		RDebug::Print(_L("BaflUtils::NearestLanguageFile (bad suffix ) profile info: %d.%03ds"), profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime/1000000, profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime%1000000);
sl@0
  1169
		}
sl@0
  1170
#endif
sl@0
  1171
	}
sl@0
  1172
	
sl@0
  1173
/** Searches for the file with the correct language extension for the language 
sl@0
  1174
of the current locale, or failing this, the best matching file.
sl@0
  1175
sl@0
  1176
@param aFs File server session.
sl@0
  1177
@param aName File name as it would be without a language-specific extension. 
sl@0
  1178
On return, this is changed to the language-specific version. If no such file 
sl@0
  1179
is found, the name is unchanged.
sl@0
  1180
@see BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage) */
sl@0
  1181
EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName)
sl@0
  1182
	{
sl@0
  1183
	TLanguage language;
sl@0
  1184
	
sl@0
  1185
	NearestLanguageFile( aFs, aName, language);
sl@0
  1186
	
sl@0
  1187
	(void)language;
sl@0
  1188
	}
sl@0
  1189
sl@0
  1190
/** Set the ideal language for the thread. 
sl@0
  1191
This interface is intended for the use of UIKON only.
sl@0
  1192
sl@0
  1193
@param aLanguage Ideal language.
sl@0
  1194
@return KErrNone, if successful; KErrNoMemory if there is not enough memory @see TLanguage
sl@0
  1195
@see BaflUtils::NearestLanguageFile() 
sl@0
  1196
@internalAll */
sl@0
  1197
EXPORT_C TInt BaflUtils::SetIdealLanguage(TLanguage aLanguage)
sl@0
  1198
	{
sl@0
  1199
	TLanguage* langPtr=(TLanguage*)Dll::Tls();
sl@0
  1200
	if( langPtr==NULL)
sl@0
  1201
		{
sl@0
  1202
		langPtr=(TLanguage*)User::Alloc(sizeof(TLanguage));
sl@0
  1203
		
sl@0
  1204
		if(!langPtr) 
sl@0
  1205
			return(KErrNoMemory);
sl@0
  1206
		
sl@0
  1207
		TInt ret=Dll::SetTls(langPtr);
sl@0
  1208
		
sl@0
  1209
		if(ret!=KErrNone)
sl@0
  1210
			return(ret);
sl@0
  1211
		}
sl@0
  1212
	*langPtr=aLanguage;
sl@0
  1213
	return(KErrNone);
sl@0
  1214
	}
sl@0
  1215
	
sl@0
  1216
/** Get the ideal language of the thread. 
sl@0
  1217
This interface is intended for the use of UIKON only.
sl@0
  1218
sl@0
  1219
@return Ideal language if set, ELangNone if not set
sl@0
  1220
@see BaflUtils::NearestLanguageFile() 
sl@0
  1221
@internalAll */
sl@0
  1222
EXPORT_C TLanguage BaflUtils::IdealLanguage()
sl@0
  1223
	{
sl@0
  1224
	TLanguage* langPtr=(TLanguage*)Dll::Tls();
sl@0
  1225
	
sl@0
  1226
	if( langPtr==NULL)
sl@0
  1227
		{
sl@0
  1228
		return(ELangNone);
sl@0
  1229
		}
sl@0
  1230
sl@0
  1231
	return(*langPtr);
sl@0
  1232
	}
sl@0
  1233
	
sl@0
  1234
EXPORT_C void BaflUtils::ReleaseIdealLanguage()
sl@0
  1235
/** Releases the ideal language store if it has been allocated. 
sl@0
  1236
This interface is intended for the use of UIKON only.
sl@0
  1237
sl@0
  1238
@internalAll */
sl@0
  1239
	{
sl@0
  1240
	TLanguage* aLanguage=(TLanguage*)Dll::Tls();
sl@0
  1241
	if( aLanguage==NULL)
sl@0
  1242
		return;
sl@0
  1243
	
sl@0
  1244
	delete aLanguage;
sl@0
  1245
	Dll::FreeTls();
sl@0
  1246
	}
sl@0
  1247
sl@0
  1248
EXPORT_C TInt BaflUtils::IsFolder(const RFs& aFs, const TDesC& aFullName, TBool& aIsFolder)
sl@0
  1249
/** Checks if the specified item is a folder.
sl@0
  1250
sl@0
  1251
@param aFs File server session
sl@0
  1252
@param aFullName Name to check
sl@0
  1253
@param aIsFolder ETrue if aFullName is a folder, otherwise EFalse
sl@0
  1254
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1255
codes */
sl@0
  1256
	{
sl@0
  1257
	TParsePtrC parse(aFullName);
sl@0
  1258
	if ((parse.DriveAndPath().Length() == KDriveAndPathLength) && (aFullName.Length() == KDriveAndPathLength))	
sl@0
  1259
		{
sl@0
  1260
		aIsFolder = ETrue;
sl@0
  1261
		return(KErrNone);
sl@0
  1262
		}
sl@0
  1263
	TEntry entry;
sl@0
  1264
	TInt retcode = aFs.Entry(aFullName, entry);
sl@0
  1265
	if (retcode == KErrNone)
sl@0
  1266
		aIsFolder = ((entry.iAtt & KEntryAttDir)==KEntryAttDir);
sl@0
  1267
sl@0
  1268
	return(retcode);
sl@0
  1269
	}
sl@0
  1270
sl@0
  1271
sl@0
  1272
EXPORT_C TBool BaflUtils::FolderExists(RFs& aFs, const TDesC& aFolderName)
sl@0
  1273
/** Tests whether a folder exists.
sl@0
  1274
sl@0
  1275
The folder is specified in a path. The path can omit the drive letter, in 
sl@0
  1276
which case the drive letter is taken from the session path.
sl@0
  1277
sl@0
  1278
If the path is badly formed, for instance if it contains illegal characters, 
sl@0
  1279
or any directory name consists of a single or double dot, or any directory 
sl@0
  1280
name includes wildcard characters, the function returns EFalse.
sl@0
  1281
sl@0
  1282
If a filename is included in the path, it is ignored (the existence 
sl@0
  1283
of the file will not be checked).  However if included, it must not
sl@0
  1284
be badly formed - this will cause the function to return EFalse.  
sl@0
  1285
If no filename is specified, the path should end in a backslash.
sl@0
  1286
sl@0
  1287
Examples of valid paths (returning ETrue):
sl@0
  1288
C:\; \; C:\Readme.txt; C:\system\data\; \system\data\Anyfile.dat
sl@0
  1289
sl@0
  1290
Examples of invalid paths (returning EFalse):
sl@0
  1291
C:\FolderDoesntExist\; ..\system\; C:\Wild*card\; C:\system\data\Bad>File.txt
sl@0
  1292
sl@0
  1293
@param aFs A connected session with the file server.
sl@0
  1294
@param aFolderName A path specifying the folder to test for.
sl@0
  1295
@return ETrue if the folder specified in aFolderName exists, EFalse if not. 
sl@0
  1296
EFalse is also returned if the specified path is badly formed. */
sl@0
  1297
	{	
sl@0
  1298
	if (aFolderName.Length()==0)
sl@0
  1299
		{return EFalse;}
sl@0
  1300
	
sl@0
  1301
	TParse parse;
sl@0
  1302
		 	
sl@0
  1303
 	TInt retcode = parse.SetNoWild(aFolderName, NULL, NULL);
sl@0
  1304
 	
sl@0
  1305
 	if (retcode != KErrNone)
sl@0
  1306
 		{return EFalse;}
sl@0
  1307
 			
sl@0
  1308
 	if (parse.NameOrExtPresent())
sl@0
  1309
 		if (!aFs.IsValidName(aFolderName))
sl@0
  1310
 			{return EFalse;}
sl@0
  1311
 	
sl@0
  1312
 	TPtrC dirName = parse.DriveAndPath();
sl@0
  1313
 	RDir dir;
sl@0
  1314
    retcode = dir.Open(aFs,dirName,0);
sl@0
  1315
 	if (retcode == KErrNone)
sl@0
  1316
 		{dir.Close();}
sl@0
  1317
 	return (retcode == KErrNone);
sl@0
  1318
	}
sl@0
  1319
sl@0
  1320
sl@0
  1321
EXPORT_C TFileName BaflUtils::FolderNameFromFullName(const TDesC& aFullName) 
sl@0
  1322
/** Gets the folder name from a path.
sl@0
  1323
sl@0
  1324
A valid full name should have a drive associated with it
sl@0
  1325
e.g ("a:\\" - "z:\\")("a:" - "z:")("c:\\system\data\file.txt")
sl@0
  1326
Invalid entry will have no drive and cause a panic EBafPanicBadOpenArg
sl@0
  1327
For example, if the path is "c:\documents\word\mydoc1", then "word" is returned.
sl@0
  1328
							"c:"						then "c:" is returned
sl@0
  1329
							"c:\\"						then "c:\" is returned
sl@0
  1330
							"c:\\mydoc1.txt				then "c:\" is returned
sl@0
  1331
sl@0
  1332
@param aFullName A path.
sl@0
  1333
@return The folder name. */
sl@0
  1334
	{
sl@0
  1335
	TParsePtrC parse(aFullName);
sl@0
  1336
	
sl@0
  1337
	__ASSERT_ALWAYS(parse.DrivePresent(),Panic(EBafPanicBadOpenArg));
sl@0
  1338
	
sl@0
  1339
	TFileName folderName = parse.Path();
sl@0
  1340
	//If the path name has no associated path(e.g "c:") or path='\'(e.g "c:\\", "c:\\file.txt")
sl@0
  1341
	//then the folder name is just equal to drivename
sl@0
  1342
	
sl@0
  1343
	TBuf<1> pathSeparator;
sl@0
  1344
	pathSeparator.Append(KPathDelimiter);
sl@0
  1345
sl@0
  1346
	if (folderName.Length()==0 || folderName==pathSeparator)
sl@0
  1347
		return (parse.DriveAndPath());
sl@0
  1348
	//else just get the foldername
sl@0
  1349
	TInt len = folderName.Length();
sl@0
  1350
	TInt pos = --len;
sl@0
  1351
	while (--pos)
sl@0
  1352
		if (folderName.Mid(pos, 1)==pathSeparator)
sl@0
  1353
			break;
sl@0
  1354
	folderName.Delete(len, 1);
sl@0
  1355
	folderName.Delete(0, pos+1);
sl@0
  1356
	return(folderName);
sl@0
  1357
	}
sl@0
  1358
sl@0
  1359
sl@0
  1360
EXPORT_C TFileName BaflUtils::DriveAndPathFromFullName(const TDesC& aFullName) 
sl@0
  1361
/** Gets the drive letter and path from a file name. 
sl@0
  1362
sl@0
  1363
This is in the form: drive-letter:\\path\\. The drive letter is folded using 
sl@0
  1364
class TCharF. 
sl@0
  1365
sl@0
  1366
@param aFullName File name
sl@0
  1367
@return The drive and path */
sl@0
  1368
	{
sl@0
  1369
	TParsePtrC parse(aFullName);
sl@0
  1370
	return (parse.DriveAndPath());
sl@0
  1371
	}
sl@0
  1372
sl@0
  1373
sl@0
  1374
EXPORT_C TFileName BaflUtils::RootFolderPath(const TBuf<1> aDriveLetter)
sl@0
  1375
/** Gets the root folder for the specified drive.
sl@0
  1376
sl@0
  1377
If aDriveLetter is an alphabet(lowercase or uppercase) then it will return
sl@0
  1378
the TFileName which is simply the drive letter plus ":\"
sl@0
  1379
If this is not the case, the function will panic with panic code EBafPanicBadOpenArg
sl@0
  1380
sl@0
  1381
@param aDriveLetter Drive letter
sl@0
  1382
@return Root folder */
sl@0
  1383
	{
sl@0
  1384
	TChar driveLetter(aDriveLetter[0]);
sl@0
  1385
	driveLetter.LowerCase();
sl@0
  1386
	TInt aDriveNumber=driveLetter-TChar('a');
sl@0
  1387
	__ASSERT_ALWAYS(aDriveNumber>= EDriveA && aDriveNumber <= EDriveZ,Panic(EBafPanicBadOpenArg));
sl@0
  1388
	
sl@0
  1389
	TFileName rootFolderPath = aDriveLetter;
sl@0
  1390
	rootFolderPath.Append(_L(":\\"));
sl@0
  1391
	return rootFolderPath;
sl@0
  1392
	}
sl@0
  1393
sl@0
  1394
sl@0
  1395
EXPORT_C void BaflUtils::AbbreviateFileName(const TFileName& aOriginalFileName, TDes& aAbbreviatedFileName)
sl@0
  1396
/** Abbreviates a file name.
sl@0
  1397
sl@0
  1398
If aOriginalFileName is less than the maximum length of aAbbreviatedFileName, 
sl@0
  1399
then the name is simply copied to aAbbreviatedFileName.
sl@0
  1400
sl@0
  1401
If this is not so, then the left-most characters of aOriginalFileName are 
sl@0
  1402
copied to aAbbreviatedFileName, up to aAbbreviatedFileName's maximum length-1. 
sl@0
  1403
aAbbreviatedFileName's first character is set to be an ellipsis.
sl@0
  1404
sl@0
  1405
@param aOriginalFileName Original file name
sl@0
  1406
@param aAbbreviatedFileName On return, abbreviated file name */
sl@0
  1407
	{
sl@0
  1408
	TInt maxWidthInChars = aAbbreviatedFileName.MaxLength();
sl@0
  1409
	if (aOriginalFileName.Length() <= maxWidthInChars)
sl@0
  1410
		{
sl@0
  1411
		aAbbreviatedFileName = aOriginalFileName;
sl@0
  1412
		return;
sl@0
  1413
		}
sl@0
  1414
	TChar ellipsis(0x2026);
sl@0
  1415
	--maxWidthInChars;  // since the ellipsis will be the first char in aAbbreviatedFileName
sl@0
  1416
	aAbbreviatedFileName.Zero();
sl@0
  1417
	aAbbreviatedFileName.Append(ellipsis);
sl@0
  1418
	aAbbreviatedFileName.Append(aOriginalFileName.Mid((aOriginalFileName.Length() - 1) - maxWidthInChars + 1, maxWidthInChars));
sl@0
  1419
	}
sl@0
  1420
sl@0
  1421
sl@0
  1422
EXPORT_C TBool BaflUtils::UidTypeMatches(const TUidType& aFileUid, const TUidType& aMatchUid)
sl@0
  1423
/** Tests whether two UID types match.
sl@0
  1424
sl@0
  1425
A match is made if each UID in aMatchUid is either identical to the corresponding 
sl@0
  1426
one in aFileUid, or is KNullUid.
sl@0
  1427
sl@0
  1428
@param aFileUid The UID type to match
sl@0
  1429
@param aMatchUid The UID type to match against
sl@0
  1430
@return ETrue if the UIDs match, otherwise EFalse */
sl@0
  1431
	{
sl@0
  1432
	for (TInt i=0; i<KMaxCheckedUid; i++)
sl@0
  1433
		{
sl@0
  1434
		if (aMatchUid[i] == KNullUid)
sl@0
  1435
			continue;
sl@0
  1436
		if (aMatchUid[i] != aFileUid[i])
sl@0
  1437
			return(EFalse);
sl@0
  1438
		}
sl@0
  1439
	return(ETrue);
sl@0
  1440
	}
sl@0
  1441
sl@0
  1442
sl@0
  1443
EXPORT_C TInt BaflUtils::Parse(const TDesC& aName)
sl@0
  1444
/** Checks if a specified file name can be parsed.
sl@0
  1445
sl@0
  1446
@param aName Name to parse
sl@0
  1447
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1448
codes */
sl@0
  1449
	{ // keeps a TParse on the stack for the minimum time possible
sl@0
  1450
	TParse parse;
sl@0
  1451
	return parse.Set(aName,NULL,NULL);
sl@0
  1452
	}
sl@0
  1453
sl@0
  1454
sl@0
  1455
EXPORT_C TInt BaflUtils::ValidateFolderNameTypedByUserL(const RFs& aFs, const TDesC& aFolderNameTypedByUser, const TDesC& aCurrentPath, TFileName& aNewFolderFullName)
sl@0
  1456
/** Checks if a folder name (without drive or path) is valid and returns the full 
sl@0
  1457
name of the folder.
sl@0
  1458
sl@0
  1459
@param aFs File server session
sl@0
  1460
@param aFolderNameTypedByUser Folder name to check
sl@0
  1461
@param aCurrentPath Path to which to add the folder
sl@0
  1462
@param aNewFolderFullName aFolderNameTypedByUser appended to aCurrentPath
sl@0
  1463
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1464
codes */
sl@0
  1465
	{
sl@0
  1466
	if (aFolderNameTypedByUser.Length() == 0)
sl@0
  1467
		return KErrArgument;	// R_EIK_TBUF_NO_FOLDERNAME_SPECIFIED;
sl@0
  1468
sl@0
  1469
	TParse* targetParse = new(ELeave) TParse;
sl@0
  1470
	CleanupStack::PushL(targetParse);
sl@0
  1471
sl@0
  1472
	TInt retcode = targetParse->Set(aFolderNameTypedByUser, NULL, NULL);
sl@0
  1473
	User::LeaveIfError(retcode);
sl@0
  1474
	if (targetParse->DrivePresent() || targetParse->PathPresent())
sl@0
  1475
		{
sl@0
  1476
		 CleanupStack::PopAndDestroy();
sl@0
  1477
		 return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
sl@0
  1478
		}
sl@0
  1479
sl@0
  1480
	if (!(aFs.IsValidName(aFolderNameTypedByUser)))
sl@0
  1481
		{
sl@0
  1482
		 CleanupStack::PopAndDestroy();
sl@0
  1483
		 return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
sl@0
  1484
		}
sl@0
  1485
		 
sl@0
  1486
sl@0
  1487
	if ((aCurrentPath.Length() + aFolderNameTypedByUser.Length() + 1) > KMaxFileName)
sl@0
  1488
		{
sl@0
  1489
		 CleanupStack::PopAndDestroy();
sl@0
  1490
		 return KErrTooBig;		//R_EIK_TBUF_FOLDERNAME_TOO_LONG;
sl@0
  1491
		}
sl@0
  1492
		 
sl@0
  1493
sl@0
  1494
	//TFileName newFolderFullName = aCurrentPath;
sl@0
  1495
	aNewFolderFullName = aCurrentPath;
sl@0
  1496
	if ((aNewFolderFullName.Length() + aFolderNameTypedByUser.Length() + 1) <= KMaxFileName)
sl@0
  1497
		{
sl@0
  1498
		aNewFolderFullName.Append(aFolderNameTypedByUser);
sl@0
  1499
		aNewFolderFullName.Append(KPathDelimiter);
sl@0
  1500
		}
sl@0
  1501
	else
sl@0
  1502
		{
sl@0
  1503
		CleanupStack::PopAndDestroy();
sl@0
  1504
		return KErrOverflow;
sl@0
  1505
		}
sl@0
  1506
sl@0
  1507
	retcode = targetParse->Set(aNewFolderFullName, NULL, NULL);
sl@0
  1508
	if (retcode != KErrNone)
sl@0
  1509
		{
sl@0
  1510
		CleanupStack::PopAndDestroy();
sl@0
  1511
		return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
sl@0
  1512
		}
sl@0
  1513
	
sl@0
  1514
	CleanupStack::PopAndDestroy();
sl@0
  1515
sl@0
  1516
	return(KErrNone);
sl@0
  1517
	}
sl@0
  1518
sl@0
  1519
void BaflUtils::DoCopyFileL(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch)
sl@0
  1520
	{
sl@0
  1521
	CFileMan* fileMan=CFileMan::NewL(aFs);
sl@0
  1522
	CleanupStack::PushL(fileMan);
sl@0
  1523
	User::LeaveIfError(fileMan->Copy(aSourceFullName,aTargetFullName,aSwitch));
sl@0
  1524
	CleanupStack::PopAndDestroy(); // fileMan
sl@0
  1525
	}
sl@0
  1526
sl@0
  1527
sl@0
  1528
EXPORT_C TInt BaflUtils::CopyFile(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch)
sl@0
  1529
/** Copies one or more files.
sl@0
  1530
sl@0
  1531
For more details, 
sl@0
  1532
@see CFileMan::Copy()
sl@0
  1533
@since     5.1
sl@0
  1534
@param     aFs File server session
sl@0
  1535
@param     aSourceFullName Path indicating the file(s) to be copied. Any path
sl@0
  1536
components that are not specified here will be taken from the session path.
sl@0
  1537
@param     aTargetFullName Path indicating the directory into which the file(s)
sl@0
  1538
are to be copied
sl@0
  1539
@param     aSwitch=CFileMan::EOverWrite Set this to zero for no overwriting and 
sl@0
  1540
no recursion; CFileMan::EOverWrite to overwrite files with the same name; or 
sl@0
  1541
CFileMan::ERecurse for recursion.
sl@0
  1542
@return   KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1543
codes.*/
sl@0
  1544
	{
sl@0
  1545
	TRAPD(err,DoCopyFileL(aFs,aSourceFullName,aTargetFullName,aSwitch));
sl@0
  1546
	return err;
sl@0
  1547
	}
sl@0
  1548
sl@0
  1549
void BaflUtils::DoDeleteFileL(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch)
sl@0
  1550
	{
sl@0
  1551
	CFileMan* fileMan=CFileMan::NewL(aFs);
sl@0
  1552
	CleanupStack::PushL(fileMan);
sl@0
  1553
	User::LeaveIfError(fileMan->Delete(aSourceFullName,aSwitch));
sl@0
  1554
	CleanupStack::PopAndDestroy(); // fileMan
sl@0
  1555
	}
sl@0
  1556
sl@0
  1557
sl@0
  1558
EXPORT_C TInt BaflUtils::DeleteFile(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch)
sl@0
  1559
/** Deletes one or more files.
sl@0
  1560
sl@0
  1561
For more details,
sl@0
  1562
@see CFileMan::Delete().
sl@0
  1563
@since 5.1
sl@0
  1564
@param aFs File server session
sl@0
  1565
@param aSourceFullName Path indicating the file(s) to be deleted. May either
sl@0
  1566
be a full path, or relative to the session path. Use wildcards to specify 
sl@0
  1567
more than one file.
sl@0
  1568
@param aSwitch=0  Specify CFileMan::ERecurse for recursion,
sl@0
  1569
zero for no recursion.
sl@0
  1570
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1571
codes. */	
sl@0
  1572
    {
sl@0
  1573
	TRAPD(err,DoDeleteFileL(aFs,aSourceFullName,aSwitch));
sl@0
  1574
	return err;
sl@0
  1575
	}
sl@0
  1576
sl@0
  1577
void BaflUtils::DoRenameFileL(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch)
sl@0
  1578
	{
sl@0
  1579
	CFileMan* fileMan=CFileMan::NewL(aFs);
sl@0
  1580
	CleanupStack::PushL(fileMan);
sl@0
  1581
	User::LeaveIfError(fileMan->Rename(aOldFullName,aNewFullName,aSwitch));
sl@0
  1582
	CleanupStack::PopAndDestroy(); // fileMan
sl@0
  1583
	}
sl@0
  1584
sl@0
  1585
sl@0
  1586
EXPORT_C TInt BaflUtils::RenameFile(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch)
sl@0
  1587
/**  Renames or moves one or more files or directories.
sl@0
  1588
sl@0
  1589
It can be used to move one or more files by specifying different
sl@0
  1590
destination and source directories.
sl@0
  1591
For more details, 
sl@0
  1592
@see CFileMan::Rename().
sl@0
  1593
@since 5.1
sl@0
  1594
@param aFs File server session
sl@0
  1595
@param aOldFullName Path specifying the file(s) to be renamed.
sl@0
  1596
@param aNewFullName Path specifying the new name for the files and/or the
sl@0
  1597
new directory. Any directories specified in this path that do not exist will 
sl@0
  1598
be created.
sl@0
  1599
@param aSwitch=CFileMan::EOverWrite  Specify zero for no overwriting, or
sl@0
  1600
CFileMan::EOverWrite to overwrite files with the same name. This 
sl@0
  1601
function cannot operate recursively.
sl@0
  1602
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1603
codes. */	
sl@0
  1604
    {
sl@0
  1605
	TRAPD(err,DoRenameFileL(aFs,aOldFullName,aNewFullName,aSwitch));
sl@0
  1606
	return err;
sl@0
  1607
	}
sl@0
  1608
sl@0
  1609
sl@0
  1610
EXPORT_C TInt BaflUtils::CheckWhetherFullNameRefersToFolder(const TDesC& aFullName, TBool& aIsFolder)
sl@0
  1611
/** Checks if a string is a valid folder name.
sl@0
  1612
sl@0
  1613
@param aFullName String to check
sl@0
  1614
@param aIsFolder ETrue if aFullName is a valid folder name, otherwise EFalse
sl@0
  1615
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1616
codes (probably because aFullName cannot be parsed). */
sl@0
  1617
	{
sl@0
  1618
	aIsFolder = EFalse;
sl@0
  1619
	TInt retcode = BaflUtils::Parse(aFullName);
sl@0
  1620
	if (retcode != KErrNone)
sl@0
  1621
		return(retcode);
sl@0
  1622
	TParsePtrC parse(aFullName);
sl@0
  1623
	if (! parse.NameOrExtPresent())
sl@0
  1624
		aIsFolder = ETrue;
sl@0
  1625
	return(KErrNone);
sl@0
  1626
	}
sl@0
  1627
sl@0
  1628
EXPORT_C TInt BaflUtils::MostSignificantPartOfFullName(const TDesC& aFullName, TFileName& aMostSignificantPart)
sl@0
  1629
/** Gets the folder name if the specified item is a valid folder name, otherwise 
sl@0
  1630
gets the file name.
sl@0
  1631
sl@0
  1632
@param aFullName Item to parse
sl@0
  1633
@param aMostSignificantPart Folder or file name
sl@0
  1634
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1635
codes */
sl@0
  1636
	{
sl@0
  1637
	TBool entryIsAFolder;
sl@0
  1638
	TInt retcode = CheckWhetherFullNameRefersToFolder(aFullName, entryIsAFolder);
sl@0
  1639
	if (retcode != KErrNone)
sl@0
  1640
		return(retcode);
sl@0
  1641
	if (entryIsAFolder)
sl@0
  1642
		{
sl@0
  1643
		aMostSignificantPart = FolderNameFromFullName(aFullName);
sl@0
  1644
		return (KErrNone);
sl@0
  1645
		}
sl@0
  1646
	// assume aFullName refers to a file
sl@0
  1647
	TParsePtrC parse(aFullName);
sl@0
  1648
	aMostSignificantPart = parse.NameAndExt();
sl@0
  1649
	return(KErrNone);
sl@0
  1650
	}
sl@0
  1651
sl@0
  1652
EXPORT_C TInt BaflUtils::CheckFolder(RFs& aFs, const TDesC& aFolderName)
sl@0
  1653
/** Checks that the specified folder can be opened.
sl@0
  1654
sl@0
  1655
@param aFs File server session
sl@0
  1656
@param aFolderName Folder to check
sl@0
  1657
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
  1658
codes */
sl@0
  1659
	{
sl@0
  1660
    RDir dir;
sl@0
  1661
    TInt retcode = dir.Open(aFs, aFolderName, 0);
sl@0
  1662
	if (retcode == KErrNone)
sl@0
  1663
		dir.Close();
sl@0
  1664
	return (retcode);
sl@0
  1665
	}
sl@0
  1666
sl@0
  1667
/**
sl@0
  1668
Checks if the specified drive is read-only.
sl@0
  1669
Checks that the KMediaAttWriteProtected and EMediaRom flags are both set.
sl@0
  1670
sl@0
  1671
@param aFs File server session
sl@0
  1672
@param aFullName File name, including drive
sl@0
  1673
@param aIsReadOnly On return, ETrue if the drive is read-only, otherwise EFalse
sl@0
  1674
@return KErrNone if successful, otherwise another of the system-wide error codes
sl@0
  1675
@see BaflUtils::DriveIsReadOnlyInternal
sl@0
  1676
*/
sl@0
  1677
EXPORT_C TInt BaflUtils::DiskIsReadOnly(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnly)
sl@0
  1678
	{
sl@0
  1679
	TInt retcode=BaflUtils::Parse(aFullName);
sl@0
  1680
	if (retcode!=KErrNone)
sl@0
  1681
		return retcode;
sl@0
  1682
	TParsePtrC parse(aFullName);
sl@0
  1683
	if (!parse.DrivePresent())
sl@0
  1684
		return KErrBadName;
sl@0
  1685
	TBuf<1> drive=parse.Drive().Left(1);
sl@0
  1686
	TChar driveLetter=drive[0];
sl@0
  1687
	TInt driveId=0;
sl@0
  1688
	retcode=RFs::CharToDrive(driveLetter,driveId);
sl@0
  1689
	if (retcode!=KErrNone)
sl@0
  1690
		return retcode;
sl@0
  1691
	TVolumeInfo volInfo;
sl@0
  1692
	retcode=aFs.Volume(volInfo,driveId);
sl@0
  1693
	if (retcode!=KErrNone)
sl@0
  1694
		return retcode;
sl@0
  1695
	aIsReadOnly=(volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected || volInfo.iDrive.iType==EMediaRom);
sl@0
  1696
	return KErrNone;
sl@0
  1697
	}
sl@0
  1698
sl@0
  1699
/** 
sl@0
  1700
Checks if the specified drive is read-only and is an internal drive i.e. non-removable.
sl@0
  1701
Checks that the KMediaAttWriteProtected and KDriveAttInternal flags are both set.
sl@0
  1702
sl@0
  1703
@param aFs File server session
sl@0
  1704
@param aFullName File name, including drive
sl@0
  1705
@param aIsReadOnlyInternal On return, ETrue if the drive is read-only and internal, otherwise EFalse
sl@0
  1706
@return KErrNone if successful, otherwise another of the system-wide errors codes 
sl@0
  1707
*/
sl@0
  1708
EXPORT_C TInt BaflUtils::DriveIsReadOnlyInternal(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnlyInternal)
sl@0
  1709
	{
sl@0
  1710
	TInt retcode=BaflUtils::Parse(aFullName);
sl@0
  1711
	if (retcode!=KErrNone)
sl@0
  1712
		return retcode;
sl@0
  1713
	TParsePtrC parse(aFullName);
sl@0
  1714
	if (!parse.DrivePresent())
sl@0
  1715
		return KErrBadName;
sl@0
  1716
	TBuf<1> drive=parse.Drive().Left(1);
sl@0
  1717
	TChar driveLetter=drive[0];
sl@0
  1718
	TInt driveId=0;
sl@0
  1719
	retcode=RFs::CharToDrive(driveLetter,driveId);
sl@0
  1720
	if (retcode!=KErrNone)
sl@0
  1721
		return retcode;
sl@0
  1722
	TVolumeInfo volInfo;
sl@0
  1723
	retcode=aFs.Volume(volInfo,driveId);
sl@0
  1724
	if (retcode!=KErrNone)
sl@0
  1725
		return retcode;
sl@0
  1726
	aIsReadOnlyInternal=((volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected)&&(volInfo.iDrive.iDriveAtt&KDriveAttInternal));
sl@0
  1727
	return KErrNone;
sl@0
  1728
	}
sl@0
  1729
sl@0
  1730
EXPORT_C void BaflUtils::GetDiskListL(const RFs& aFs,CDesCArray& aArray)
sl@0
  1731
/** Retrieves a list of all drives on the system.
sl@0
  1732
sl@0
  1733
The file server is interrogated for a list of the drive letters for all available 
sl@0
  1734
drives. 
sl@0
  1735
sl@0
  1736
On emulator:
sl@0
  1737
The removable media is represented by drive X: .
sl@0
  1738
sl@0
  1739
On hardware:
sl@0
  1740
The removable media is represented by drives D: E: F: and G: .
sl@0
  1741
sl@0
  1742
@param aFs A connected session with the file server.
sl@0
  1743
@param aArray On return, contains the drive letters that correspond to the 
sl@0
  1744
available drives. The drive letters are uppercase and are in alphabetical 
sl@0
  1745
order. */
sl@0
  1746
	{ // static
sl@0
  1747
	aArray.Reset();
sl@0
  1748
	TDriveList driveList;
sl@0
  1749
	User::LeaveIfError(aFs.DriveList(driveList));
sl@0
  1750
	for (TInt ii=0;ii<KMaxDrives;ii++)
sl@0
  1751
		{
sl@0
  1752
		if (driveList[ii])
sl@0
  1753
			{
sl@0
  1754
			TChar drive;
sl@0
  1755
			User::LeaveIfError(aFs.DriveToChar(ii,drive));
sl@0
  1756
			drive.UpperCase();
sl@0
  1757
			TBuf<1> buf;
sl@0
  1758
			buf.Append(drive);
sl@0
  1759
			aArray.AppendL(buf);
sl@0
  1760
			}
sl@0
  1761
		}
sl@0
  1762
	}
sl@0
  1763
sl@0
  1764
EXPORT_C void BaflUtils::UpdateDiskListL(const RFs& aFs,CDesCArray& aArray,TBool aIncludeRom,TDriveNumber aDriveNumber)
sl@0
  1765
/** Retrieves a list of all drives present on the system.
sl@0
  1766
sl@0
  1767
The file server is interrogated for a list of the drive letters for all available 
sl@0
  1768
drives. The drive letter that corresponds to aDriveNumber is added to the 
sl@0
  1769
list regardless of whether it is present, or is corrupt. Also, the C: drive 
sl@0
  1770
is forced onto the list, even if corrupt or not present.
sl@0
  1771
sl@0
  1772
On emulator:
sl@0
  1773
The removable media is represented by drive X: and is forced onto the list 
sl@0
  1774
unless removed (F5,F4).
sl@0
  1775
sl@0
  1776
On hardware:
sl@0
  1777
The removable media is represented by drives D: E: F: and G: and is forced 
sl@0
  1778
onto the list regardless of whether it is present, or is corrupt.
sl@0
  1779
sl@0
  1780
@param aFs A connected session with the file server.
sl@0
  1781
@param aArray On return, contains the drive letters that correspond to the 
sl@0
  1782
available drives. The drive letters are uppercase and are in alphabetical 
sl@0
  1783
order.
sl@0
  1784
@param aIncludeRom Specify ETrue if the ROM drive should be included in the 
sl@0
  1785
list, EFalse if not.
sl@0
  1786
@param aDriveNumber The drive to force into the list, e.g. the drive in the 
sl@0
  1787
default path. */
sl@0
  1788
	{ // static
sl@0
  1789
	aArray.Reset();
sl@0
  1790
	TDriveList driveList;
sl@0
  1791
	User::LeaveIfError(aFs.DriveList(driveList));
sl@0
  1792
	for (TInt ii=0;ii<KMaxDrives;ii++)
sl@0
  1793
		{
sl@0
  1794
		if (driveList[ii] || ii==aDriveNumber)
sl@0
  1795
			{
sl@0
  1796
			TVolumeInfo vInfo;
sl@0
  1797
			const TInt err=aFs.Volume(vInfo,TDriveUnit(ii));
sl@0
  1798
			if (err==KErrNone || err==KErrCorrupt || ii==aDriveNumber || BaflUtils::IsFirstDriveForSocket(TDriveUnit(ii)))
sl@0
  1799
				{
sl@0
  1800
				if (ii==EDriveZ && vInfo.iDrive.iDriveAtt&KDriveAttRom && !aIncludeRom)
sl@0
  1801
					continue;
sl@0
  1802
				TChar drive;
sl@0
  1803
				User::LeaveIfError(aFs.DriveToChar(ii,drive));
sl@0
  1804
				drive.UpperCase();
sl@0
  1805
				TBuf<1> buf;
sl@0
  1806
				buf.Append(drive);
sl@0
  1807
				aArray.AppendL(buf);
sl@0
  1808
				}
sl@0
  1809
			}
sl@0
  1810
		}
sl@0
  1811
	}
sl@0
  1812
sl@0
  1813
EXPORT_C TBool BaflUtils::IsFirstDriveForSocket(TDriveUnit aDriveUnit)
sl@0
  1814
/** Tests whether the specified drive corresponds to the primary partition in a 
sl@0
  1815
removable media slot.
sl@0
  1816
sl@0
  1817
The function assumes that the D: drive corresponds to the primary partition 
sl@0
  1818
on socket 0, and that the E: drive corresponds to the primary partition on 
sl@0
  1819
socket 1 (a socket is a slot for removable media). This mapping may not always 
sl@0
  1820
be the case because it is set up in the variant layer of the Symbian OS.
sl@0
  1821
sl@0
  1822
This function assumes that the drive mappings are contiguous, starting 
sl@0
  1823
from drive D: .
sl@0
  1824
sl@0
  1825
On emulator:
sl@0
  1826
The removable media is represented by drive X: only.
sl@0
  1827
sl@0
  1828
@param aDriveUnit The drive to check.
sl@0
  1829
@return ETrue if the drive is the primary partition in a removable media slot. 
sl@0
  1830
ETrue is also returned if the drive is A, B or C:. EFalse is returned otherwise. */
sl@0
  1831
	{ // static
sl@0
  1832
	TDriveInfoV1Buf buf;
sl@0
  1833
	UserHal::DriveInfo(buf);
sl@0
  1834
sl@0
  1835
#ifdef __EPOC32__
sl@0
  1836
	return ((aDriveUnit-EDriveC)<=buf().iTotalSockets);
sl@0
  1837
#else // emulator
sl@0
  1838
    return (aDriveUnit==EDriveX || (aDriveUnit-EDriveC)<=buf().iTotalSockets);
sl@0
  1839
#endif
sl@0
  1840
	}
sl@0
  1841
sl@0
  1842
EXPORT_C void BaflUtils::RemoveSystemDirectory(CDir& aDir)
sl@0
  1843
/** Removes "System" from a list of directory entries. 
sl@0
  1844
sl@0
  1845
@param aDir Array of directory entries. */
sl@0
  1846
	{ // static
sl@0
  1847
	STATIC_CAST(BaflDir&,aDir).RemoveSystem();
sl@0
  1848
	}
sl@0
  1849
sl@0
  1850
EXPORT_C TInt BaflUtils::SortByTable(CDir& aDir,CBaflFileSortTable* aTable)
sl@0
  1851
/** Sorts files by UID.
sl@0
  1852
sl@0
  1853
The caller supplies a table which specifies the order in which files are to 
sl@0
  1854
be sorted. The files whose UID3 is the first UID in the table appear first. 
sl@0
  1855
The files whose UID3 is the UID specified second appear next, and so on. Files 
sl@0
  1856
whose UID3 is not specified in the table, and directories, appear at the end 
sl@0
  1857
of the list, with directories preceding the files, and with files sorted in 
sl@0
  1858
ascending order of UID3.
sl@0
  1859
sl@0
  1860
This function is used for customising how lists of application files are sorted.
sl@0
  1861
sl@0
  1862
@param aDir The array of files and directories to sort.
sl@0
  1863
@param aTable A sort order table containing the UIDs to use in the sort.
sl@0
  1864
@return KErrNone if successful, otherwise one of the standard error codes. */
sl@0
  1865
	{ // static
sl@0
  1866
	return STATIC_CAST(BaflDir&,aDir).SortByTable(aTable);
sl@0
  1867
	}
sl@0
  1868
sl@0
  1869
EXPORT_C void BaflUtils::GetDowngradePathL(const RFs& aFs, const TLanguage aCurrentLanguage, RArray<TLanguage>& aLanguageArray)
sl@0
  1870
/** Gets the full language downgrade path for a particular locale.
sl@0
  1871
sl@0
  1872
@param aFs A connected session with the file server.
sl@0
  1873
@param aCurrentLanguage The language of the locale for which the language downgrade 
sl@0
  1874
path is required. This language will always be returned as the first language 
sl@0
  1875
in aLanguageArray. To get the downgrade path for the language of the current 
sl@0
  1876
locale, specify the language returned by User::Language(). 
sl@0
  1877
@param aLanguageArray On return, contains the language downgrade path.
sl@0
  1878
@see BaflUtils::NearestLanguageFile() */
sl@0
  1879
 	{
sl@0
  1880
 	TLocale currentLocale; 
sl@0
  1881
  	TNearestLanguageFileFinder languageDowngradePath(aFs);
sl@0
  1882
  	TLanguage idealLanguage=IdealLanguage();
sl@0
  1883
  	MakeLanguageDowngradePath(languageDowngradePath.iPath,aCurrentLanguage,idealLanguage, currentLocale);
sl@0
  1884
 	aLanguageArray.Reset();
sl@0
  1885
  	const TLanguage* p=languageDowngradePath.iPath;
sl@0
  1886
  	while (*p != ELangNone)
sl@0
  1887
  		{
sl@0
  1888
		User::LeaveIfError(aLanguageArray.Append(*p));
sl@0
  1889
  		++p;
sl@0
  1890
  		}
sl@0
  1891
 	}
sl@0
  1892
sl@0
  1893
EXPORT_C void BaflUtils::PersistLocale()
sl@0
  1894
/** Saves the locale settings in TLocale and the currency symbol to file. 
sl@0
  1895
@deprecated 9.1
sl@0
  1896
Persistence and initialisation of system locale data will be performed 
sl@0
  1897
transparently by a separate executable (InilialiseLocale.exe) wich should 
sl@0
  1898
be loaded as part of the system start procedure.
sl@0
  1899
*/
sl@0
  1900
	{
sl@0
  1901
// Replaced by new repository based locale initialisation mechanism.
sl@0
  1902
	}
sl@0
  1903
sl@0
  1904
sl@0
  1905
sl@0
  1906
EXPORT_C TInt BaflUtils::PersistHAL()
sl@0
  1907
/** Saves the HAL settings to file. 
sl@0
  1908
This will start a new executable and saves HAL attributes to a file, 
sl@0
  1909
little delay because of the creation of new process
sl@0
  1910
@return KErrNone if suceessful, otheriwse system wide error code.
sl@0
  1911
*/
sl@0
  1912
	{
sl@0
  1913
	RProcess process;
sl@0
  1914
	_LIT(KHALSettings, "HALSettings.exe");
sl@0
  1915
	_LIT(KCommandLine, "PERSIST");
sl@0
  1916
	TInt result = process.Create(KHALSettings, KCommandLine);
sl@0
  1917
	if(result != KErrNone )
sl@0
  1918
		return result;
sl@0
  1919
	TRequestStatus status;
sl@0
  1920
	process.Logon(status);
sl@0
  1921
	if ( status != KRequestPending)
sl@0
  1922
		{
sl@0
  1923
		process.Kill(0);		// abort 
sl@0
  1924
		}
sl@0
  1925
	else
sl@0
  1926
		{
sl@0
  1927
		process.Resume();	// logon OK
sl@0
  1928
		}
sl@0
  1929
	User::WaitForRequest(status);
sl@0
  1930
	
sl@0
  1931
	// we can't use the 'exit reason' if the exe panicked as this
sl@0
  1932
	// is the panic 'reason' and may be '0' which cannot be distinguished
sl@0
  1933
	// from KErrNone
sl@0
  1934
	result = process.ExitType() == EExitPanic ? KErrGeneral : status.Int();
sl@0
  1935
	process.Close();
sl@0
  1936
	return result;
sl@0
  1937
	}
sl@0
  1938
sl@0
  1939
EXPORT_C void BaflUtils::PersistScreenCalibration(const TDigitizerCalibration& aScreenCalibration)
sl@0
  1940
	{
sl@0
  1941
	
sl@0
  1942
	RFs	fs;
sl@0
  1943
	TInt err = fs.Connect();
sl@0
  1944
	if (err == KErrNone)
sl@0
  1945
		{
sl@0
  1946
		// Setting up drive to store Screen data
sl@0
  1947
		TDriveUnit systemDrive(static_cast<TInt>(RFs::GetSystemDrive()));
sl@0
  1948
		TBuf<KMaxDriveName+KScreenCalibrationPathLength>  ScreenFileNameWithDriveAndPath(systemDrive.Name());
sl@0
  1949
		ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFolder);
sl@0
  1950
		
sl@0
  1951
		// Ensure directory \System\Data exists in target drive
sl@0
  1952
		TRAP(err, EnsurePathExistsL(fs, ScreenFileNameWithDriveAndPath));
sl@0
  1953
		if(err == KErrNone)
sl@0
  1954
			{
sl@0
  1955
			ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFileName);	
sl@0
  1956
			
sl@0
  1957
			RFile file;
sl@0
  1958
			err = file.Replace(fs,ScreenFileNameWithDriveAndPath,EFileWrite|EFileShareExclusive);
sl@0
  1959
			if (err == KErrNone)
sl@0
  1960
				{
sl@0
  1961
				// Write aScreenCalibration to file. 
sl@0
  1962
				TPtrC8 calptr((const TUint8*)&aScreenCalibration, sizeof(TDigitizerCalibration));
sl@0
  1963
				err = file.Write(calptr);
sl@0
  1964
				}
sl@0
  1965
			file.Close();
sl@0
  1966
			}
sl@0
  1967
		}
sl@0
  1968
	fs.Close();
sl@0
  1969
	}
sl@0
  1970
sl@0
  1971
EXPORT_C void BaflUtils::InitialiseScreenCalibration(RFs& aFs)
sl@0
  1972
	{
sl@0
  1973
	TFindFile ff(aFs);
sl@0
  1974
	if (ff.FindByDir(KScreenCalibrationFileName, KScreenCalibrationFolder)==KErrNone)
sl@0
  1975
		{
sl@0
  1976
		RFile file; 
sl@0
  1977
		if (file.Open(aFs,ff.File(),EFileRead) == KErrNone )
sl@0
  1978
			{
sl@0
  1979
			TDigitizerCalibration calibrationSetting;
sl@0
  1980
			TPtr8 scrcal((TUint8*)&calibrationSetting, sizeof(TDigitizerCalibration));
sl@0
  1981
			if( file.Read(scrcal, sizeof( TDigitizerCalibration )) == KErrNone )
sl@0
  1982
				UserHal::SetXYInputCalibration(calibrationSetting);
sl@0
  1983
sl@0
  1984
			}
sl@0
  1985
		file.Close();
sl@0
  1986
		}
sl@0
  1987
	}
sl@0
  1988
sl@0
  1989
EXPORT_C void BaflUtils::InitialiseHAL(RFs&)
sl@0
  1990
/** Initialise the HAL settings from. 
sl@0
  1991
@deprecated 9.1
sl@0
  1992
This function is empty
sl@0
  1993
*/
sl@0
  1994
	{
sl@0
  1995
	}
sl@0
  1996
sl@0
  1997
EXPORT_C void BaflUtils::InitialiseLocale(RFs& /* aFs */) 
sl@0
  1998
	{
sl@0
  1999
// Replaced by new repository based locale initialisation mechanism.
sl@0
  2000
	}
sl@0
  2001
sl@0
  2002
sl@0
  2003
//
sl@0
  2004
// class CEikFileSortTable
sl@0
  2005
// 
sl@0
  2006
sl@0
  2007
/**
sl@0
  2008
*/	
sl@0
  2009
EXPORT_C CBaflFileSortTable::CBaflFileSortTable()
sl@0
  2010
	: CArrayFixFlat<TUid>(EArrayGranularity)
sl@0
  2011
	{}
sl@0
  2012
sl@0
  2013
/**
sl@0
  2014
 * Loads the CBaflFileSortTable using the UIDs read from the TResourceReader supplied in aReader. 
sl@0
  2015
 * @param aReader TResourceReader from which UIDS are read.
sl@0
  2016
 * @leave KErrNoMemory if there is insufficient memory available or one of the system wide error codes.
sl@0
  2017
 */	
sl@0
  2018
EXPORT_C void CBaflFileSortTable::ConstructFromResourceL(TResourceReader& aReader)
sl@0
  2019
	{
sl@0
  2020
	const TInt count=aReader.ReadInt16();
sl@0
  2021
	for (TInt i=0;i<count;i++)
sl@0
  2022
		AppendL(TUid::Uid(aReader.ReadInt32()));
sl@0
  2023
	}
sl@0
  2024
sl@0
  2025
//
sl@0
  2026
// class BaflDir
sl@0
  2027
//
sl@0
  2028
sl@0
  2029
#define KSystemDirName _L("System") // Name for System directory
sl@0
  2030
sl@0
  2031
sl@0
  2032
void BaflDir::RemoveSystem()
sl@0
  2033
/*  Remove "System" if in list and it's a directory. */	
sl@0
  2034
{
sl@0
  2035
	TInt index;
sl@0
  2036
	TEntry entry;
sl@0
  2037
	entry.iName=KSystemDirName;
sl@0
  2038
	TKeyArrayPak key(_FOFF(TEntry,iName),ECmpFolded);
sl@0
  2039
	if (iArray->Find(entry,key,index)==KErrNone)
sl@0
  2040
		{
sl@0
  2041
		entry=(*iArray)[index];
sl@0
  2042
		if (entry.IsDir())
sl@0
  2043
			iArray->Delete(index);
sl@0
  2044
		}
sl@0
  2045
	};
sl@0
  2046
sl@0
  2047
/*
sl@0
  2048
This function gets the element at position "aPos" of aEntries array and inserts 
sl@0
  2049
it to poition "aNewPos". The element size is "aSize". After the operation the array size
sl@0
  2050
grows by 1, the element at "aPos" position  moves one position forward.
sl@0
  2051
This function must be called only from BaflDir::SortByTable() and the insert position
sl@0
  2052
is always less or equal than the position of the element to be inserted.
sl@0
  2053
*/
sl@0
  2054
static void InsertL(CArrayPakFlat<TEntry>* aEntries, TInt aPos, TInt aNewPos, TInt aSize)
sl@0
  2055
	{
sl@0
  2056
	__ASSERT_DEBUG(aPos >= aNewPos, User::Invariant());
sl@0
  2057
	//Expand the array adding one empty entry at "aNewPos" position.
sl@0
  2058
	aEntries->ExpandL(aNewPos, aSize);
sl@0
  2059
	//After successfull "ExpandL" call "aPos" must be incremented by 1.
sl@0
  2060
	//Copy the entry from "aPos + 1" position to "aNewPos" position
sl@0
  2061
	(*aEntries)[aNewPos] = (*aEntries)[++aPos];
sl@0
  2062
	}
sl@0
  2063
sl@0
  2064
/**
sl@0
  2065
Copied from f32file.inl (now moved to f32file_private.h)
sl@0
  2066
Returns the minimum uncompressed size of the TEntry object, including the valid 
sl@0
  2067
portion of the name string. The returned value is aligned to 4-byte boundary and
sl@0
  2068
length includes private members.
sl@0
  2069
sl@0
  2070
@internalTechnology
sl@0
  2071
@return minimum uncompressed size of TEntry object
sl@0
  2072
*/
sl@0
  2073
sl@0
  2074
TInt BaflDir::MinEntrySize(const TEntry & aEntry)
sl@0
  2075
	{
sl@0
  2076
	return(sizeof(TUint)+sizeof(TInt)+sizeof(TTime)+sizeof(TInt)+sizeof(TUidType)+
sl@0
  2077
         Align4(aEntry.iName.Size()) + 2*sizeof(TInt));
sl@0
  2078
	}
sl@0
  2079
sl@0
  2080
TInt BaflDir::SortByTable(CBaflFileSortTable* aTable)
sl@0
  2081
/**
sl@0
  2082
Sort into order from given table.
sl@0
  2083
Any file with uid[2] matching an entry in the table will be sorted relative to 
sl@0
  2084
others in the table and before any files with no matching uid. 
sl@0
  2085
For Example: Assume UID table is filled with below 2 UID's
sl@0
  2086
table[0] =0x10003a64 and table[1] =0x10003a5c. Then file with UID[2]=0x10003a64 
sl@0
  2087
will be sorted first in the list followed by file with UID[2]=0x10003a5c. Rest
sl@0
  2088
files will be sorted in the ascending order of UID[2] with directories preceding
sl@0
  2089
the files.
sl@0
  2090
sl@0
  2091
@param aTable A sort order table containing the UIDs to use in the sort. 
sl@0
  2092
@return KErrNone if suceessful, otheriwse another system-wide error code.
sl@0
  2093
*/
sl@0
  2094
	{
sl@0
  2095
	TInt r=this->Sort(EDirsFirst|ESortByUid);
sl@0
  2096
	if(r != KErrNone)
sl@0
  2097
		{
sl@0
  2098
		return r;
sl@0
  2099
		}
sl@0
  2100
	const TInt tableCount=aTable->Count();
sl@0
  2101
	const TInt count=iArray->Count();
sl@0
  2102
	TInt sortedInsertionPoint = 0;
sl@0
  2103
	for (TInt i=0;i<tableCount;i++)	
sl@0
  2104
		{
sl@0
  2105
		TUid	tableUid;
sl@0
  2106
		// get each UID in the table
sl@0
  2107
		tableUid = aTable->At(i);
sl@0
  2108
		for (TInt j=sortedInsertionPoint;j<count;j++)
sl@0
  2109
			{
sl@0
  2110
			//parse files in the array list
sl@0
  2111
			TEntry* pEntry=&(*iArray)[j];
sl@0
  2112
			// check table UID for match with UID[2] of the file in the list
sl@0
  2113
			// if found, move the file at the top in the list, followed by next matching UID
sl@0
  2114
			if (tableUid == pEntry->iType[2])
sl@0
  2115
				{
sl@0
  2116
				TRAPD(insertErr, ::InsertL(iArray, j, sortedInsertionPoint++, MinEntrySize(*pEntry)));
sl@0
  2117
				if(insertErr!=KErrNone)
sl@0
  2118
					{
sl@0
  2119
					return insertErr;
sl@0
  2120
					}
sl@0
  2121
				iArray->Delete(j+1);
sl@0
  2122
				}
sl@0
  2123
			}
sl@0
  2124
		}
sl@0
  2125
		iArray->Compress();
sl@0
  2126
		return KErrNone;
sl@0
  2127
	}