os/ossrv/lowlevellibsandfws/apputils/src/BAUTILS.CPP
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/lowlevellibsandfws/apputils/src/BAUTILS.CPP	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,2127 @@
     1.4 +// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     1.5 +// All rights reserved.
     1.6 +// This component and the accompanying materials are made available
     1.7 +// under the terms of "Eclipse Public License v1.0"
     1.8 +// which accompanies this distribution, and is available
     1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
    1.10 +//
    1.11 +// Initial Contributors:
    1.12 +// Nokia Corporation - initial contribution.
    1.13 +//
    1.14 +// Contributors:
    1.15 +//
    1.16 +// Description:
    1.17 +// Started by DWW, November 1995
    1.18 +// BAFL utilities static class
    1.19 +//
    1.20 +//
    1.21 +
    1.22 +#include "BaUtilsImp.h"
    1.23 +#include <e32hal.h>
    1.24 +#include <bautils.h>
    1.25 +#include <baflpan.h>
    1.26 +#include <baliba.h>
    1.27 +#include <hal.h>
    1.28 +#include <hal_data.h>
    1.29 +#include <utf.h>
    1.30 +
    1.31 +/**
    1.32 +Mimimum length of a filename and mimimum length of a suffix.
    1.33 +Note these two values are tied together.
    1.34 +*/
    1.35 +const TInt KInvNameAndMinSuffixLength = 2;
    1.36 +
    1.37 +#define ISDIGIT(c) (c-'0' >= 0 && c-'0' <= 9)
    1.38 +
    1.39 +_LIT(KAllDrives, "YXWVUTSRQPONMLKJIHGFEDCBAZ");
    1.40 +const TInt KDriveAndPathLength = 3;
    1.41 +
    1.42 +// screen calibration stuff
    1.43 +_LIT(KScreenCalibrationFolder,"\\System\\Data\\");
    1.44 +_LIT(KScreenCalibrationFileName, "Screen.DAT");
    1.45 +const TInt KScreenCalibrationPathLength = 23;	// folder + filename
    1.46 +
    1.47 +
    1.48 +// #define DO_PROFILING
    1.49 +
    1.50 +#if defined(DO_PROFILING)
    1.51 +#pragma message ("------------ N.B. profiling of \"BaflUtils::NearestLanguageFile\" is enabled")
    1.52 +#include <e32svr.h>
    1.53 +#define FIRST_PROFILE_INDEX	50
    1.54 +#define PROFILE_INDEX_1		(FIRST_PROFILE_INDEX+0)
    1.55 +#define PROFILE_COUNT		1
    1.56 +#endif
    1.57 +
    1.58 +// BaflUtils
    1.59 +
    1.60 +class BaflDir : public CDir
    1.61 +	{
    1.62 +public:
    1.63 +	void RemoveSystem();
    1.64 +	TInt SortByTable(CBaflFileSortTable* aTable);
    1.65 +private:
    1.66 +	TInt MinEntrySize(const TEntry & aEntry);
    1.67 +	};
    1.68 +
    1.69 +
    1.70 +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 };
    1.71 +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 };
    1.72 +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 };
    1.73 +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 };
    1.74 +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 };
    1.75 +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 };
    1.76 +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 };
    1.77 +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 };
    1.78 +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 };
    1.79 +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 };
    1.80 +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 };
    1.81 +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 };
    1.82 +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 };
    1.83 +LOCAL_C const TLanguage dp13[] = { ELangInternationalFrench,ELangFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
    1.84 +LOCAL_C const TLanguage dp14[] = { ELangBelgianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangCanadianFrench,ELangNone };
    1.85 +LOCAL_C const TLanguage dp15[] = { ELangCanadianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangNone };
    1.86 +LOCAL_C const TLanguage dp16[] = { ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
    1.87 +LOCAL_C const TLanguage dp17[] = { ELangSwissFrench,ELangFrench,ELangInternationalFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone };
    1.88 +LOCAL_C const TLanguage dp18[] = { ELangSwissGerman,ELangGerman,ELangAustrian,ELangNone };
    1.89 +LOCAL_C const TLanguage dp19[] = { ELangAustrian,ELangGerman,ELangSwissGerman,ELangNone };
    1.90 +LOCAL_C const TLanguage dp20[] = { ELangGerman,ELangSwissGerman,ELangAustrian,ELangNone };
    1.91 +LOCAL_C const TLanguage dp21[] = { ELangSerbian,ELangCroatian,ELangNone };
    1.92 +LOCAL_C const TLanguage dp22[] = { ELangCroatian,ELangSerbian,ELangNone };
    1.93 +LOCAL_C const TLanguage dp23[] = { ELangRomanian,ELangMoldavian,ELangNone };
    1.94 +LOCAL_C const TLanguage dp24[] = { ELangMoldavian,ELangRomanian,ELangNone };
    1.95 +LOCAL_C const TLanguage dp25[] = { ELangBelgianFlemish,ELangDutch,ELangNone };
    1.96 +LOCAL_C const TLanguage dp26[] = { ELangDutch,ELangBelgianFlemish,ELangNone };
    1.97 +LOCAL_C const TLanguage dp27[] = { ELangAfrikaans,ELangDutch,ELangNone };
    1.98 +LOCAL_C const TLanguage dp28[] = { ELangMalay_Apac,ELangMalay,ELangNone };
    1.99 +LOCAL_C const TLanguage dp29[] = { ELangIndonesian_Apac,ELangIndonesian,ELangNone };
   1.100 +LOCAL_C const TLanguage dp30[] = { ELangSpanish,ELangInternationalSpanish,ELangLatinAmericanSpanish,ELangNone };
   1.101 +LOCAL_C const TLanguage dp31[] = { ELangLatinAmericanSpanish,ELangSpanish,ELangInternationalSpanish,ELangNone };
   1.102 +LOCAL_C const TLanguage dp32[] = { ELangInternationalSpanish,ELangSpanish,ELangLatinAmericanSpanish,ELangNone };
   1.103 +LOCAL_C const TLanguage dp33[] = { ELangCyprusGreek,ELangGreek,ELangNone };
   1.104 +LOCAL_C const TLanguage dp34[] = { ELangGreek,ELangCyprusGreek,ELangNone };
   1.105 +LOCAL_C const TLanguage dp35[] = { ELangSwissItalian,ELangItalian,ELangNone };
   1.106 +LOCAL_C const TLanguage dp36[] = { ELangItalian,ELangSwissItalian,ELangNone };
   1.107 +LOCAL_C const TLanguage dp37[] = { ELangBrazilianPortuguese,ELangPortuguese,ELangNone };
   1.108 +LOCAL_C const TLanguage dp38[] = { ELangPortuguese,ELangBrazilianPortuguese,ELangNone };
   1.109 +LOCAL_C const TLanguage dp39[] = { ELangFinlandSwedish,ELangSwedish,ELangNone };
   1.110 +LOCAL_C const TLanguage dp40[] = { ELangSwedish,ELangFinlandSwedish,ELangNone };
   1.111 +LOCAL_C const TLanguage dp41[] = { ELangCyprusTurkish,ELangTurkish,ELangNone };
   1.112 +LOCAL_C const TLanguage dp42[] = { ELangTurkish,ELangCyprusTurkish,ELangNone };
   1.113 +LOCAL_C const TLanguage dp43[] = { ELangHongKongChinese, ELangTaiwanChinese, ELangPrcChinese,ELangNone };
   1.114 +LOCAL_C const TLanguage dp44[] = { ELangTaiwanChinese, ELangHongKongChinese,ELangPrcChinese,ELangNone };
   1.115 +LOCAL_C const TLanguage dp45[] = { ELangPrcChinese, ELangHongKongChinese, ELangTaiwanChinese,ELangNone };
   1.116 +LOCAL_C const TLanguage * const KEquivalentLists[] = { dp0,  dp1,  dp2,  dp3,  dp4,  dp5,  dp6,  
   1.117 +		dp7,  dp8,  dp9,  dp10,  dp11,  dp12,  dp13,  dp14,  dp15,  dp16,  dp17,
   1.118 +		dp18,  dp19,  dp20,  dp21,  dp22,  dp23,  dp24,  dp25,  dp26,  dp27,  
   1.119 +		dp28,  dp29,  dp30,  dp31,  dp32,  dp33,  dp34,  dp35,  dp36,  dp37,  
   1.120 +		dp38,  dp39,  dp40,  dp41,  dp42,  dp43,  dp44,  dp45};
   1.121 +
   1.122 +/**
   1.123 +This function gets the list of languages that are 'equivalent' to the
   1.124 +given language. We say language L1 is equivalent to language L2 if 
   1.125 +speakers of L2 can readily understand L1 without intentional study 
   1.126 +or extraordinary effort.
   1.127 +
   1.128 +The equivalence relationship is defined in a static table. Please refer 
   1.129 +to the definition of 'KEquivalentLists' for details.
   1.130 +Each row in the table is formatted like this:
   1.131 +@code
   1.132 +L1, L2, L3, ..., Ln, ELangNone
   1.133 +@codeend
   1.134 +In which L2, ..., Ln are equivalents of L1, and ELangNone marks the end of an
   1.135 +entry. The list is ordered. Compared with L3, L2 is nearer to L1. When choosing
   1.136 +an equivalent of L1, L2 shall be preferred over L3, L3 shall be preferred 
   1.137 +over L4, and so on.  
   1.138 +L1 is always returned as the ‘nearest equivalent’ of L1 itself.
   1.139 +
   1.140 +BaflUtils::NearestLanguageFileV2 searches language specific resource files 
   1.141 +according to the 'equivalent' relationship returned by this function.
   1.142 + 
   1.143 +@param aLang The language whose equivalents needs to be found out.
   1.144 +@param aEquivalents On return, this array contains the ordered list of 
   1.145 +       languages that are equivalent to the given language. If there is no 
   1.146 +       entry for the given language in the table, this array will contain 
   1.147 +       two elements on return: the first is the given language itself 
   1.148 +       and the 2nd one is ELangNone. For any language that has equivalents 
   1.149 +       defined, content of he corresponding entry is returned.     
   1.150 +       
   1.151 +@see BaflUtils::NearestLanguageFileV2
   1.152 +*/ 
   1.153 +EXPORT_C void 
   1.154 +BaflUtils::GetEquivalentLanguageList(TLanguage aLang, TLanguagePath& aEquivalents) 
   1.155 +	{
   1.156 +	aEquivalents[0] = aLang;
   1.157 +	aEquivalents[1] = ELangNone;
   1.158 +	const TInt len = sizeof(KEquivalentLists) / sizeof(KEquivalentLists[0]);
   1.159 +	for (TInt i = 0; i < len; ++i)
   1.160 +		{
   1.161 +		const TLanguage *ptr = KEquivalentLists[i];
   1.162 +		if (ptr[0] == aLang)
   1.163 +			{
   1.164 +			TInt index = 1;
   1.165 +			while (ELangNone != *ptr)
   1.166 +				{
   1.167 +				aEquivalents[index++] = (TLanguage)*(++ptr);
   1.168 +				}
   1.169 +			aEquivalents[index] = ELangNone;
   1.170 +			break;
   1.171 +			} // end if ptr[0]
   1.172 +		} // end for i
   1.173 +	}
   1.174 +
   1.175 +/**
   1.176 +NearestLanguageFileV2 is very similar to the existing 'NearestLanguageFile'
   1.177 +function. The only difference between NearestLanguageFile and
   1.178 +NearestLanguageFileV2 is the order in which language specific 
   1.179 +resource files are searched for. 
   1.180 +NearestLanguageFile searches language specific resource files in the 
   1.181 +order defined by the 'downgrade path' of the given language. Content of the 
   1.182 +downgrade path is dependent on the current active locale, and parts of 
   1.183 +it is runtime configurable.
   1.184 +NearestLanguageFileV2 searches for language specific resource files 
   1.185 +in the order defined by the 'language equivalence table', which is a  
   1.186 +static data table fixed at build time. There is one entry in the table for 
   1.187 +each language that has one or more equivalents.
   1.188 +
   1.189 +@param aFs An active file server session.
   1.190 +@param aName Name of the language-neutral resource file name which consist of
   1.191 +an optional drive specification, followed by an optional path name,
   1.192 +followed by basename for filename, followed by a period and extension.
   1.193 +On return, in case of a match, this is replaced by the language-specific version
   1.194 +which consists of the last two characters of the extension plus any preceding
   1.195 +numeric characters being replaced by the language code. Remains unchanged when there's no match 
   1.196 +@param aLanguage On return, in case of a match, this is replaced by the corresponding language.
   1.197 +  In case of no match, it is set to ELangNone.  
   1.198 +
   1.199 +@see TLanguage
   1.200 +@see BaflUtils::GetEquivalentLanguageList
   1.201 + */
   1.202 +EXPORT_C void 
   1.203 +BaflUtils::NearestLanguageFileV2(const RFs& aFs,TFileName& aName, TLanguage& aLanguage)
   1.204 +	{
   1.205 +	TNearestLanguageFileFinder finder(aFs);
   1.206 +	TBool goodSuffix=finder.SetFileName(aName);
   1.207 +	
   1.208 +	// Continue only if the suffix is good.
   1.209 +	if(goodSuffix)
   1.210 +		{
   1.211 +		// add preset customised resource drive to drive list  
   1.212 +		// Note that errors returned from AddCustomResourceDrive are ignored. This is because if 
   1.213 +		// a custom resource drive has not been found we still want to continue on with searching 
   1.214 +		// other drives according to our algorithm
   1.215 +		finder.AddCustomResourceDrive();
   1.216 +		
   1.217 +		GetEquivalentLanguageList(User::Language(), finder.iPath);
   1.218 +		if (!finder.FindLanguageAndDrive()
   1.219 +			&& KErrNone != finder.FindFirstLanguageFileAndDrive())
   1.220 +			finder.RepairFileName();
   1.221 +		aLanguage = finder.Language();
   1.222 +		}
   1.223 +	else
   1.224 +		{
   1.225 +		aLanguage = ELangNone;
   1.226 +		}
   1.227 +	}
   1.228 +
   1.229 +// TLibAssocBase
   1.230 +
   1.231 +EXPORT_C TLibAssocBase::TLibAssocBase(const RLibrary& aLib,TAny* aPtr)
   1.232 +	: iLibrary(aLib),iPtr(aPtr)
   1.233 +/**
   1.234 +Constructs the object taking the specified DLL and a class instance.
   1.235 +
   1.236 +@param aLib A reference to a DLL that has already been opened.
   1.237 +@param aPtr An untyped pointer to an object to be associated with the DLL.
   1.238 +            Typically, this object will have been created using
   1.239 +            the ordinal 1 function from that DLL. */	
   1.240 +	{}
   1.241 +
   1.242 +
   1.243 +
   1.244 +
   1.245 +EXPORT_C void TLibAssocBase::Set(const RLibrary& aLib,TAny* aPtr)
   1.246 +/**
   1.247 +Implements TLibAssoc::Set().
   1.248 +
   1.249 +@param aLib   A reference to a DLL that has already been opened.
   1.250 +@param aClass A pointer to an object to be associated with the DLL.
   1.251 +              Typically, this object will have been created using
   1.252 +              the ordinal 1 function from that DLL.
   1.253 +
   1.254 +@see TLibAssoc::Set */
   1.255 +	{
   1.256 +	__ASSERT_ALWAYS(iLibrary.Handle()==KNullHandle&&iPtr==NULL,Panic(EBafPanicLibAssocAlreadySet));
   1.257 +	iLibrary=aLib;
   1.258 +	iPtr=aPtr;
   1.259 +	}
   1.260 +
   1.261 +
   1.262 +
   1.263 +
   1.264 +EXPORT_C void TLibAssocBase::DoUnload(TAny* aThis)
   1.265 +/**
   1.266 +Calls Close() on the associated DLL.
   1.267 +
   1.268 +@param aThis An untyped pointer to a TLibAssoc type.
   1.269 +*/
   1.270 +	{
   1.271 +	TLibAssocBase& l=*(TLibAssocBase*)aThis;
   1.272 +	l.iPtr=NULL;
   1.273 +	l.iLibrary.Close();
   1.274 +	}
   1.275 +
   1.276 +//
   1.277 +// class BaflUtils
   1.278 +//
   1.279 +EXPORT_C void BaflUtils::CopyWithTruncation(TDes& aDest,const TDesC& aSrc,TChar aTruncationSymbol)
   1.280 +/** Copies a descriptor, abbreviating it to fit the destination descriptor.
   1.281 +
   1.282 +If aSrc is less than the maximum length of aDest, then the string is simply 
   1.283 +copied to aDest.
   1.284 +
   1.285 +If this is not so, then the left-most characters of aSrc are copied to aDest, 
   1.286 +up to aDest's maximum length-1. aDest's final character is set to be aTruncationSymbol.
   1.287 +
   1.288 +@param aDest On return, the truncated string
   1.289 +@param aSrc The string to truncate
   1.290 +@param aTruncationSymbol The truncation character to add */
   1.291 +	{ // static
   1.292 +	TInt maxLength=aDest.MaxLength();
   1.293 +	if (aSrc.Length()<=maxLength)
   1.294 +		aDest.Copy(aSrc);
   1.295 +	else
   1.296 +		{
   1.297 +		aDest.Copy(aSrc.Left(maxLength-1));
   1.298 +		aDest.Append(aTruncationSymbol);
   1.299 +		}
   1.300 +	}
   1.301 +
   1.302 +EXPORT_C TBool BaflUtils::FileExists(const RFs& aFileSession,const TDesC& aFileName)
   1.303 +/** Checks if the specified file exists.
   1.304 +	
   1.305 +@param aFs File server session
   1.306 +@param aFileName File to check
   1.307 +@return ETrue if the file exists, otherwise EFalse */
   1.308 +	{ // static
   1.309 +	TEntry entry;
   1.310 +	return(aFileSession.Entry(aFileName,entry)==KErrNone);
   1.311 +	}
   1.312 +
   1.313 +EXPORT_C TBool BaflUtils::PathExists(RFs& aFs,const TDesC& aPath)
   1.314 +/** Tests whether a path exists.
   1.315 +
   1.316 +The path should contain a drive letter and a directory, or EFalse is returned. 
   1.317 +EFalse is also returned if it contains a filename or filename extension.
   1.318 +
   1.319 +If the path is badly formed, for instance if it contains illegal characters, 
   1.320 +or any directory name consists of a single or double dot, or any directory 
   1.321 +name includes wildcard characters, the function returns EFalse.
   1.322 +
   1.323 +@param aFs A connected session with the file server.
   1.324 +@param aPath The path to test for. It should end in a backslash.
   1.325 +@return ETrue if the path exists, EFalse if not. EFalse is also returned if the 
   1.326 +specified path is badly formed. */
   1.327 +	{ // static
   1.328 +	TParse parse;
   1.329 +	TInt retcode;
   1.330 +	retcode = parse.Set(aPath, NULL, NULL);
   1.331 +	if (retcode != KErrNone)
   1.332 +		return EFalse;
   1.333 +	if ((! parse.DrivePresent()) || (parse.NameOrExtPresent()))
   1.334 +		return EFalse;
   1.335 +	if (parse.Path().Length() == 0)
   1.336 +		return EFalse;
   1.337 +	TFileName dirName = parse.DriveAndPath();
   1.338 +	if (dirName.Length() > KMaxFileName)
   1.339 +		return(EFalse);
   1.340 +    RDir dir;
   1.341 +    retcode = dir.Open(aFs,dirName,0);
   1.342 +	if (retcode == KErrNone)
   1.343 +		dir.Close();
   1.344 +	return (retcode == KErrNone);
   1.345 +	}
   1.346 +
   1.347 +EXPORT_C void BaflUtils::EnsurePathExistsL(RFs& aFileSession,const TDesC& aFileName)
   1.348 +/** Makes one or more directories, if they do not already exist. 
   1.349 +	
   1.350 +Any valid path component in the specified path that does not already exist 
   1.351 +is created as a directory. If the specified path already exists, the function 
   1.352 +returns normally.
   1.353 +
   1.354 +@param aFs File server session
   1.355 +@param aFileName Path to ensure exists
   1.356 +@see RFs::MkDirAll() */
   1.357 +	{ // static
   1.358 +	TInt error=aFileSession.MkDirAll(aFileName);
   1.359 +	if (error!=KErrAlreadyExists)
   1.360 +		User::LeaveIfError(error);
   1.361 +	}
   1.362 +
   1.363 +EXPORT_C TPtrC BaflUtils::ExtractAppNameFromFullName(const TFullName &aName)
   1.364 +/** Gets the application name from a full thread name.
   1.365 +
   1.366 +@param aName Thread name
   1.367 +@return Application name
   1.368 +@see RThread */
   1.369 +	{
   1.370 +	// static - return the app name (after first :: before next ::, if any) from a full thread name
   1.371 +	TChar delimiter=':';
   1.372 +	TInt start=aName.Locate(delimiter);
   1.373 +	if (start<0)
   1.374 +		start=0; // should never happen
   1.375 +	else if (aName.Length()>start+2)
   1.376 +		start+=2;
   1.377 +	TPtrC rest=aName.Mid(start);
   1.378 +	TInt end=rest.Locate(delimiter);
   1.379 +	return end<0 ? rest : rest.Left(end);
   1.380 +	}
   1.381 +
   1.382 +LOCAL_C TBool IsLanguageExtended(const TLanguage aLanguage)
   1.383 +	{
   1.384 +	// For compatibility reasons, ELangNone is 0xFFFF. However, it's not an extended language.
   1.385 +	if ((aLanguage==ELangNone) || ((static_cast<TUint>(aLanguage))<=KDialectMask))
   1.386 +		return EFalse;
   1.387 +	else
   1.388 +		return ETrue;
   1.389 +	}
   1.390 +
   1.391 +
   1.392 +LOCAL_C TLanguage BaseLanguage(const TLanguage aLanguage)
   1.393 +	{
   1.394 +	if (IsLanguageExtended(aLanguage))
   1.395 +		return static_cast<TLanguage>(aLanguage & KDialectMask);
   1.396 +	else
   1.397 +		return aLanguage;
   1.398 +	}
   1.399 +
   1.400 +LOCAL_C TLanguage NextLanguage(TLanguage aLanguage)
   1.401 +/** Returns the next best language to use after aLanguage,
   1.402 +based on Symbian's base table of language near-equivalence.
   1.403 +@internalAll */
   1.404 +	{
   1.405 +	switch (aLanguage)
   1.406 +		{
   1.407 +		case ELangAustralian:
   1.408 +		case ELangNewZealand:
   1.409 +		case ELangSouthAfricanEnglish:
   1.410 +		case ELangInternationalEnglish:
   1.411 +		case ELangAmerican:
   1.412 +		case ELangEnglish_Apac:
   1.413 +		case ELangEnglish_Taiwan:
   1.414 +		case ELangEnglish_HongKong:
   1.415 +		case ELangEnglish_Prc:
   1.416 +		case ELangEnglish_Japan:
   1.417 +		case ELangEnglish_Thailand:
   1.418 +			return ELangEnglish;
   1.419 +		case ELangCanadianEnglish:
   1.420 +			return ELangAmerican;	// 2-stage downgrade
   1.421 +		case ELangSwissFrench:
   1.422 +		case ELangBelgianFrench:
   1.423 +		case ELangInternationalFrench:
   1.424 +		case ELangCanadianFrench:
   1.425 +			return ELangFrench;
   1.426 +		case ELangSwissGerman:
   1.427 +		case ELangAustrian:
   1.428 +			return ELangGerman;
   1.429 +		case ELangInternationalSpanish:
   1.430 +		case ELangLatinAmericanSpanish:
   1.431 +			return ELangSpanish;
   1.432 +		case ELangSwissItalian:
   1.433 +			return ELangItalian;
   1.434 +		case ELangFinlandSwedish:
   1.435 +			return ELangSwedish;
   1.436 +		case ELangCyprusTurkish:
   1.437 +			return ELangTurkish;
   1.438 +		case ELangBelgianFlemish:
   1.439 +			return ELangDutch;
   1.440 +		case ELangHongKongChinese:
   1.441 +			return ELangTaiwanChinese;
   1.442 +		case ELangCyprusGreek:
   1.443 +			return ELangGreek;
   1.444 +		case ELangMalay_Apac:
   1.445 +			return ELangMalay;
   1.446 +		case ELangBrazilianPortuguese:
   1.447 +			return ELangPortuguese;
   1.448 +		default:
   1.449 +			return ELangNone;	
   1.450 +		}
   1.451 +	}
   1.452 +
   1.453 +
   1.454 +void AddLanguage(TLanguagePath& aPath, TLanguage aNewLanguage)
   1.455 +/** Add language to the language path if there is space.
   1.456 +The first empty slot must have "ELangNone" in it. This will also be true
   1.457 +on exit. */	
   1.458 +	{
   1.459 +	TLanguage *p = aPath;
   1.460 +	const TLanguage *end = &(aPath[KMaxDowngradeLanguages]);
   1.461 +	while (p != end)
   1.462 +		{
   1.463 +		if (*p == aNewLanguage)
   1.464 +			// language already in list
   1.465 +			break;
   1.466 +		if (*p == ELangNone)
   1.467 +			{
   1.468 +			// found the end of the list
   1.469 +			p[0] = aNewLanguage;
   1.470 +			p[1] = ELangNone;
   1.471 +			break;
   1.472 +			}
   1.473 +		++p;
   1.474 +		}
   1.475 +	return;
   1.476 +	}
   1.477 +
   1.478 +void MakeLanguageDowngradePath(TLanguagePath& aPath,
   1.479 +	TLanguage aCurrent, TLanguage aIdeal, const TLocale& aLocale)
   1.480 +	{
   1.481 +	TInt j = 0;
   1.482 +	if( aIdeal != ELangNone)
   1.483 +		{
   1.484 +		aPath[j++]=aIdeal;	
   1.485 +		}
   1.486 +	aPath[j++] = aCurrent;
   1.487 +	aPath[j++] = ELangNone;
   1.488 +
   1.489 +	if (aCurrent & ~KDialectMask)
   1.490 +		AddLanguage(aPath, static_cast<TLanguage>(aCurrent & KDialectMask));
   1.491 +
   1.492 +	for (TInt i=0;i<=2;i++)
   1.493 +		{
   1.494 +		AddLanguage(aPath, aLocale.LanguageDowngrade(i));
   1.495 +		AddLanguage(aPath, BaseLanguage(aLocale.LanguageDowngrade(i)));
   1.496 +		}
   1.497 +
   1.498 +	while (ELangNone != (aCurrent = NextLanguage(BaseLanguage(aCurrent))))  
   1.499 +		AddLanguage(aPath, aCurrent);
   1.500 +	}
   1.501 +
   1.502 +TInt RRealDirectoryScanner::Open(RFs& aFs, const TDesC& aMatchPattern)
   1.503 +	{
   1.504 +	return iDir.Open(aFs, aMatchPattern,
   1.505 +		KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive);
   1.506 +	}
   1.507 +
   1.508 +TInt RRealDirectoryScanner::Next(TEntry& aOut)
   1.509 +	{
   1.510 +	return iDir.Read(aOut);
   1.511 +	}
   1.512 +
   1.513 +void RRealDirectoryScanner::Close()
   1.514 +	{
   1.515 +	iDir.Close();
   1.516 +	}
   1.517 +
   1.518 +/**
   1.519 +Simply counts the number of numerical characters at the end of the name passed.
   1.520 +
   1.521 +@internalComponent
   1.522 +@param			aFilename The filename to parse
   1.523 +				
   1.524 +@return			Count of the numeric digits at the end of the name passed, 
   1.525 +				e.g. x.r491 gives 3.
   1.526 +*/
   1.527 +TInt TNearestLanguageFileFinder::CountDigitsFromEnd(const TDesC& aFilename)
   1.528 +	{
   1.529 +	TInt digitCount = 0;
   1.530 +	
   1.531 +	for (TInt idx = aFilename.Length()-1; idx>=0 && ISDIGIT (aFilename [idx]); --idx)
   1.532 +		{
   1.533 +		++digitCount;
   1.534 +		}
   1.535 +		
   1.536 +	return digitCount;
   1.537 +	}
   1.538 +
   1.539 +
   1.540 +/**
   1.541 +Counts the number of digits at the end of a filename.
   1.542 +
   1.543 +@internalComponent
   1.544 +@param			aFilename The filename to parse
   1.545 +				
   1.546 +@return			Count of the numeric digits at the end of the suffix, 
   1.547 +				e.g. x.r491 gives 3.
   1.548 +				0 if no numeric end of suffix,
   1.549 +				KErrBadName for an invalid filename,
   1.550 +				KErrNotSupported if the filename (minus path) is less
   1.551 +				than or equal to KInvNameAndMinSuffixLength in length
   1.552 +*/
   1.553 +TInt TNearestLanguageFileFinder::CountDigitsFromEndInSuffix (const TDesC& aFilename)
   1.554 +	{
   1.555 +	TInt digitCount = 0;
   1.556 +	TInt slashIdx = 0;
   1.557 +	TInt len = aFilename.Length ();
   1.558 +	
   1.559 +	// NOTE: We didn't use TChar here as they are too slow.
   1.560 +	// We also didn't use TParse as they are too large.
   1.561 +	
   1.562 +	// don't work on the path
   1.563 +	for (slashIdx=len-1; slashIdx >= 0 && aFilename[slashIdx] != '\\'; --slashIdx)
   1.564 +	{/*do nothing*/};
   1.565 +	
   1.566 +	// Get new length
   1.567 +	if (slashIdx>=0) {len = len-slashIdx-1;}
   1.568 +
   1.569 +	// Initial test to see if filename legal size.
   1.570 +	if (len > KInvNameAndMinSuffixLength)
   1.571 +		{
   1.572 +		digitCount = CountDigitsFromEnd(aFilename);
   1.573 +
   1.574 +		// Can't store something bigger or we'll panic!
   1.575 +		if (digitCount > KMaxSuffixLength)
   1.576 +			{
   1.577 +			digitCount = KErrBadName;
   1.578 +			}
   1.579 +		else
   1.580 +		// numeric filename, e.g. "1234". 
   1.581 +		// No preceeding alpha character
   1.582 +		if (!(len-digitCount))
   1.583 +			{
   1.584 +			digitCount = KErrBadName;
   1.585 +			}
   1.586 +		}
   1.587 +	else
   1.588 +		{
   1.589 +		digitCount = KErrNotSupported;
   1.590 +		}
   1.591 +		
   1.592 +	return digitCount;
   1.593 +	}
   1.594 +
   1.595 +RDirectoryScanner& TNearestLanguageFileFinder::DirectoryScanner()
   1.596 +	{
   1.597 +	return iDirScanner;
   1.598 +	}
   1.599 +
   1.600 +TBool TNearestLanguageFileFinder::FileExists(const TDesC& aFileName) const
   1.601 +	{
   1.602 +	return BaflUtils::FileExists(iFs, aFileName);
   1.603 +	}
   1.604 +
   1.605 +TBool TNearestLanguageFileFinder::FindDrive()
   1.606 +	{
   1.607 +	ASSERT(iFileName);
   1.608 +	TBool found=EFalse;
   1.609 +	TInt driveLength=iDrives.Length();
   1.610 +	for (TInt drive = 0; drive!=driveLength; ++drive)
   1.611 +		{
   1.612 +		(*iFileName)[0] = iDrives[drive];
   1.613 +		if (FileExists(*iFileName))
   1.614 +			{
   1.615 +			found=ETrue;
   1.616 +			break;
   1.617 +			}
   1.618 +		}
   1.619 +	return found;
   1.620 +	}
   1.621 +
   1.622 +TBool TNearestLanguageFileFinder::AppendLanguageCode(TLanguage aLanguage)
   1.623 +	{
   1.624 +	TInt rest = static_cast<TInt>(aLanguage);
   1.625 +#ifdef _DEBUG
   1.626 +	_LIT(KErrorMessage, "Bafl");
   1.627 +#endif
   1.628 +	__ASSERT_DEBUG(0 <= rest, User::Panic(KErrorMessage,KErrArgument));
   1.629 +	iFileName->SetLength(iBaseLength);
   1.630 +	const TInt remaining = iFileName->MaxLength() - iBaseLength;
   1.631 +	TInt soFar = 0;
   1.632 +	TBuf<1> num;
   1.633 +	num.Append('0');
   1.634 +	TBool appendLangSuccess = ETrue;
   1.635 +	TInt digitCount = 0;
   1.636 +	TInt digit = 0;
   1.637 +	while (rest)
   1.638 +		{
   1.639 +		if (remaining == soFar)
   1.640 +			{
   1.641 +			// no more room in descriptor- return rather than panic,
   1.642 +			// file cannot exist.
   1.643 +			iFileName->SetLength(iBaseLength);
   1.644 +			appendLangSuccess= EFalse;
   1.645 +			break;
   1.646 +			}
   1.647 +		// Convert the number to ASCII by consistantly getting the base 10 remainder to convert.
   1.648 +		// The number is updated minus the remainder for the next iteration.
   1.649 +		// eg (rest = 123) -> (12, r3) -> (1, r2) -> (0, r1)
   1.650 +		// Then insert the ASCII representation of the remainder into the filename end
   1.651 +		// so it appears the correct way round.
   1.652 +		// eg (filename.r) -> (filename.r3) -> (filename.r23) -> (filename.r123)
   1.653 +		digit = rest % 10;
   1.654 +		digitCount++;
   1.655 +		rest /= 10;
   1.656 +		num[0] = static_cast<TText16>(digit + '0');
   1.657 +		iFileName->Insert(iBaseLength, num);
   1.658 +
   1.659 +		// Minimum suffix length is KInvNameAndMinSuffixLength
   1.660 +		// so we have to insert zeros to make this up.
   1.661 +		while (!rest && digitCount < KInvNameAndMinSuffixLength)
   1.662 +			{
   1.663 +			num[0] = static_cast<TText16>('0');
   1.664 +			iFileName->Insert(iBaseLength, num);
   1.665 +			++digitCount;
   1.666 +			}
   1.667 +			
   1.668 +		++soFar;
   1.669 +		}
   1.670 +		
   1.671 +	return appendLangSuccess;
   1.672 +	}
   1.673 +
   1.674 +
   1.675 +TBool TNearestLanguageFileFinder::FindLanguageAndDrive()
   1.676 +/** Search for files across all drives in all languages in the path plus the
   1.677 +language-neutral file. */
   1.678 +	{
   1.679 +	ASSERT(iFileName);
   1.680 +	// No point appending if the suffix is bad
   1.681 +	for (const TLanguage* currentLang = iPath; *currentLang != ELangNone; ++currentLang)
   1.682 +		{
   1.683 +		if (AppendLanguageCode(*currentLang) && FindDrive())
   1.684 +			{
   1.685 +			iLanguage = *currentLang;
   1.686 +			return ETrue;
   1.687 +			}
   1.688 +		}
   1.689 +	// search for language-neutral file
   1.690 +	iFileName->SetLength(iBaseLength);
   1.691 +	iFileName->Append(iSuffix);
   1.692 +	return FindDrive();
   1.693 +	}
   1.694 +
   1.695 +TInt TNearestLanguageFileFinder::LanguageNumberFromFile(const TDesC& aFileName, const TDesC& aStem)
   1.696 +	{
   1.697 +	TInt lang = 0;
   1.698 +	TInt multiplier = 1;
   1.699 +	TInt leadingZeroCount = 0;
   1.700 +	TInt languageNumber = KErrNotFound;
   1.701 +	const TText* firstChar = aFileName.Ptr();
   1.702 +	const TText* lastChar = firstChar + aFileName.Length() - 1;
   1.703 +	const TText* currentChar = lastChar;
   1.704 +	// string cannot contain only numbers, because it must have a ':' in it
   1.705 +	while ('0' <= *currentChar && *currentChar <= '9')
   1.706 +		{
   1.707 +		if (*currentChar == '0')
   1.708 +			leadingZeroCount++;
   1.709 +		else
   1.710 +			{
   1.711 +			leadingZeroCount = 0;
   1.712 +			lang += multiplier * (*currentChar - '0');
   1.713 +			}
   1.714 +		multiplier *= 10;
   1.715 +		--currentChar;
   1.716 +		}
   1.717 +	TInt along=lastChar - currentChar;
   1.718 +	if (2 <= along)
   1.719 +		{
   1.720 +		// We have at least 2 digits at the end.
   1.721 +		// trim of bad leading zeros
   1.722 +		TInt maxTrim = along - 2;
   1.723 +		if (maxTrim < leadingZeroCount)
   1.724 +			{
   1.725 +			leadingZeroCount = maxTrim;
   1.726 +			}
   1.727 +		currentChar += leadingZeroCount;
   1.728 +		// we have at least 2 digits at the end but does the rest of it match the stem?
   1.729 +		TPtrC foundStem(firstChar, currentChar - firstChar + 1);
   1.730 +		//foundStem.CompareF(aStem.Right(foundStem.Length()))
   1.731 +		if (0 == foundStem.CompareF(aStem))
   1.732 +			{
   1.733 +			languageNumber=lang;
   1.734 +			}
   1.735 +		}
   1.736 +	return languageNumber;
   1.737 +	}
   1.738 +
   1.739 +TInt TNearestLanguageFileFinder::FindFirstLanguageFile(RFs& aFs)
   1.740 +	{
   1.741 +	ASSERT(iFileName);
   1.742 +	iFileName->SetLength(iBaseLength);
   1.743 +	TPtrC name(*iFileName);
   1.744 +	TParsePtrC nameToParse(name);
   1.745 +	TPtrC nameStem(nameToParse.NameAndExt());
   1.746 +	iFileName->Append('*');
   1.747 +	TInt bestLanguageMatch = KMaxTInt;
   1.748 +	RDirectoryScanner& scanner = DirectoryScanner();
   1.749 +	TInt err = scanner.Open(aFs, *iFileName);
   1.750 +	if (err != KErrNone)
   1.751 +		{
   1.752 +		return err;
   1.753 +		}
   1.754 +	TEntry entry;
   1.755 +	while (KErrNone == scanner.Next(entry))
   1.756 +		{
   1.757 +		TInt lang = LanguageNumberFromFile(entry.iName, nameStem);
   1.758 +		if (0 < lang && lang < bestLanguageMatch)
   1.759 +			{
   1.760 +			bestLanguageMatch = lang;
   1.761 +			}
   1.762 +		}
   1.763 +	scanner.Close();
   1.764 +	if (bestLanguageMatch != KMaxTInt)
   1.765 +		{
   1.766 +		iLanguage = static_cast<TLanguage>(bestLanguageMatch);
   1.767 +		AppendLanguageCode(static_cast<TLanguage>(bestLanguageMatch));
   1.768 +		return KErrNone;
   1.769 +		}
   1.770 +	return KErrNotFound;
   1.771 +	}
   1.772 +
   1.773 +// Try each drive for any language files
   1.774 +// iFileName must have a directory specifier
   1.775 +TInt TNearestLanguageFileFinder::FindFirstLanguageFileAndDrive()
   1.776 +	{
   1.777 +	ASSERT(iFileName);
   1.778 +	TInt findFirstResult=KErrNotFound;
   1.779 +	TInt driveLength=iDrives.Length();
   1.780 +	for (TInt drive = 0; drive != driveLength; ++drive)
   1.781 +		{
   1.782 +		(*iFileName)[0] = iDrives[drive];
   1.783 +		TInt err = FindFirstLanguageFile(CONST_CAST(RFs&,iFs));
   1.784 +		if (err == KErrNone || err == KErrNoMemory)
   1.785 +			{
   1.786 +			findFirstResult=err;
   1.787 +			break;
   1.788 +			}
   1.789 +		}
   1.790 +	return findFirstResult;
   1.791 +	}
   1.792 +
   1.793 +/**
   1.794 +Invalid filenames are any filename whose length (minus path) must be greater
   1.795 +than KInvNameAndMinSuffixLength, and whose form is purely numerical, i.e. '1234' 
   1.796 +*/
   1.797 +TBool TNearestLanguageFileFinder::SetFileName(TFileName& aFileName)
   1.798 +	{
   1.799 +	iDrives.Zero();
   1.800 +	iFileName = &aFileName;
   1.801 +	iOriginalBaseLength = iFileName->Length();
   1.802 +	
   1.803 +	TInt suffixLength = CountDigitsFromEndInSuffix (aFileName);
   1.804 +	
   1.805 +	// No point trying for filenames thats are badly formed
   1.806 +	// or that are too large.
   1.807 +	if (suffixLength >= 0 && 
   1.808 +		KInvNameAndMinSuffixLength < iOriginalBaseLength)
   1.809 +		{
   1.810 +		if (suffixLength > 0)
   1.811 +			{
   1.812 +			// all of suffix to be replaced 
   1.813 +			iSuffix = iFileName->Right(suffixLength);
   1.814 +			iOriginalBaseLength -= suffixLength;
   1.815 +			iFileName->SetLength(iOriginalBaseLength);
   1.816 +			}
   1.817 +		else
   1.818 +			{ 
   1.819 +			// No numerical part to suffix
   1.820 +			TInt periodIdx = 0;
   1.821 +			
   1.822 +			// Search for the period within range KInvNameAndMinSuffixLength 
   1.823 +			// from the end. As this must work for all values of
   1.824 +			// KInvNameAndMinSuffixLength
   1.825 +			for (TInt i = iOriginalBaseLength-1; 
   1.826 +			     !periodIdx && i >= (iOriginalBaseLength-KInvNameAndMinSuffixLength-1);
   1.827 +			     --i)
   1.828 +				{
   1.829 +				if ((*iFileName) [i] == '.')
   1.830 +					{
   1.831 +					periodIdx = i;
   1.832 +					}
   1.833 +				}
   1.834 +			
   1.835 +			// Don't handle files ending in a period.
   1.836 +			// This is because the behaviour is different between Windows
   1.837 +			// and Symbian Fs. In Windows it strips the period off.
   1.838 +			//
   1.839 +			// However, and this shouldn't happen as it is not shown
   1.840 +			// (in the documentation) to be valid.
   1.841 +			// Just try our best.
   1.842 +			if (periodIdx == iOriginalBaseLength-1)
   1.843 +				{
   1.844 +				iSuffix.Zero();
   1.845 +				return EFalse;
   1.846 +				}
   1.847 +			else
   1.848 +			if (periodIdx)
   1.849 +				{
   1.850 +				// If there are KInvNameAndMinSuffixLength chars after the period
   1.851 +				// simply replace them.
   1.852 +				TInt right = iOriginalBaseLength-periodIdx-1;
   1.853 +				iSuffix = iFileName->Right(right);
   1.854 +				iOriginalBaseLength -= right;
   1.855 +				iFileName->SetLength(iOriginalBaseLength);					
   1.856 +				}
   1.857 +			else
   1.858 +				{
   1.859 +				// Make the suffix start from KInvNameAndMinSuffixLength 
   1.860 +				// from the right
   1.861 +				TInt right = KInvNameAndMinSuffixLength;
   1.862 +				iSuffix = iFileName->Right(right);
   1.863 +				iOriginalBaseLength -= right;
   1.864 +				iFileName->SetLength(iOriginalBaseLength);					
   1.865 +				}
   1.866 +			}
   1.867 +		}
   1.868 +	else
   1.869 +		{
   1.870 +		// bad or no suffix - treat the same
   1.871 +		iSuffix.Zero();
   1.872 +		return EFalse;
   1.873 +		}
   1.874 +
   1.875 +	// For filenames with no drive letter prefix and also for filenames
   1.876 +	// shorter than the drive letter length, i.e. with no drive
   1.877 +	// information, insert it.
   1.878 +	// Handles if the user simply enters the drive, e.g. "c:".
   1.879 +	if (iOriginalBaseLength < KMaxDriveName || (*iFileName)[1] != ':')
   1.880 +		{
   1.881 +		// Set up the default if none supplied and make room in the filename 
   1.882 +        // array to contain a drive specification. Set initial drive letter to -1
   1.883 +        // so the iFileName is repaired before exited 
   1.884 +		iInitialDriveLetter = -1;
   1.885 +		iFileName->Insert(0, _L("_:")); 
   1.886 +		iDrives.Append('Z');
   1.887 +		}
   1.888 +	else
   1.889 +		{
   1.890 +	   // Use the drive supplied inthe aName to NearestLanguageFile()
   1.891 +		iInitialDriveLetter = (*iFileName)[0];
   1.892 +		iDrives.Append(iInitialDriveLetter);
   1.893 +		}
   1.894 +	
   1.895 +	iBaseLength = iFileName->Length();
   1.896 +	
   1.897 +	return ETrue;
   1.898 +	}
   1.899 +
   1.900 +
   1.901 +TLanguage TNearestLanguageFileFinder::Language()
   1.902 +	{
   1.903 +	return iLanguage;
   1.904 +	}
   1.905 +
   1.906 +TNearestLanguageFileFinder::TNearestLanguageFileFinder(
   1.907 +	const RFs& aFs)
   1.908 +	: iFs(aFs), iFileName(0), iLanguage(ELangNone)
   1.909 +	{
   1.910 +	}
   1.911 +
   1.912 +void TNearestLanguageFileFinder::RepairFileName()
   1.913 +	{
   1.914 +	ASSERT(iFileName);
   1.915 +	iFileName->SetLength(iBaseLength);
   1.916 +	if (iInitialDriveLetter == -1)
   1.917 +		iFileName->Delete(0, 2);
   1.918 +	else
   1.919 +		(*iFileName)[0] = static_cast<TText>(iInitialDriveLetter);
   1.920 +	iFileName->SetLength(iOriginalBaseLength);
   1.921 +	iFileName->Append(iSuffix);
   1.922 +	}
   1.923 +
   1.924 +
   1.925 +/**
   1.926 +Add the custom resource drive to the start of the iDrives string.
   1.927 +
   1.928 +The custom resource drive is a preset writeable drive on which customised 
   1.929 +resource files may be present. This drive takes priority over the other 
   1.930 +drives when searching for language files.
   1.931 +
   1.932 +@return KErrNone if iDrives string was successfully modified; KErrAlreadyExists 
   1.933 +if the drive is already present in the string; otherwise one of 
   1.934 +the other system-wide error codes (iDrives will be unmodified). 
   1.935 +*/
   1.936 +TInt TNearestLanguageFileFinder::AddCustomResourceDrive()
   1.937 +	{
   1.938 +	TInt drive = GetCustomResourceDriveNumber();
   1.939 +	if (drive<0)
   1.940 +		return drive;
   1.941 +	
   1.942 +	// if drive not already in drive list
   1.943 +	if (iDrives.LocateF('A' + drive) < 0)
   1.944 +		{
   1.945 +		// add it
   1.946 +		_LIT(KDrivePlaceholder, "_");
   1.947 +		iDrives.Insert(0, KDrivePlaceholder);
   1.948 +		iDrives[0] = 'A' + drive;
   1.949 +		return KErrNone;
   1.950 +		}
   1.951 +	else
   1.952 +		return KErrAlreadyExists;
   1.953 +	}
   1.954 +
   1.955 +
   1.956 +void TNearestLanguageFileFinder::AddAllDrives()
   1.957 +	{
   1.958 +	ASSERT(iDrives.Length() < 2);
   1.959 +	if (iDrives.Length() == 0)
   1.960 +		{
   1.961 +		iDrives = KAllDrives;
   1.962 +		return;
   1.963 +		}
   1.964 +	TInt pos = KAllDrives().LocateF(iDrives[0]);
   1.965 +	if (pos < 0)
   1.966 +		{
   1.967 +		iDrives = KAllDrives;
   1.968 +		return;
   1.969 +		}
   1.970 +	iDrives.Append(KAllDrives().Left(pos));
   1.971 +	iDrives.Append(KAllDrives().Mid(pos + 1));
   1.972 +	}
   1.973 +
   1.974 +
   1.975 +/**
   1.976 +Get the value of the custom resource drive.
   1.977 +
   1.978 +The custom resource drive is a preset writeable drive on which customised language resource 
   1.979 +files can reside. The drive number is accessed via the HAL attribute ECustomResourceDrive. 
   1.980 +It is then returned if it has been defined as a valid drive no.
   1.981 +Otherwise for backward compatibility reasons an attempt is then made to access the system 
   1.982 +drive HAL attribute instead. This drive number is returned if it has been defined as a valid 
   1.983 +drive number.  
   1.984 +Otherwise if neither a valid ECustomResourceDrive or ESystemDrive exists then KErrNotFound 
   1.985 +is returned.
   1.986 + 
   1.987 +Note that the ESystemDrive HAL attribute has been deprecated. It is accessed here to cater 
   1.988 +for existing implementations which still expect it to be used.
   1.989 + 
   1.990 +@return The drive number (corresponding to a TDriveNumber value) if successful; 
   1.991 +KErrNotFound if neither a valid ECustomResourceDrive or a valid ESystemDrive HAL attribute 
   1.992 +is defined;  
   1.993 + 
   1.994 +@see HAL::ECustomResourceDrive
   1.995 +@see HAL::ESystemDrive
   1.996 +*/
   1.997 +TInt TNearestLanguageFileFinder::GetCustomResourceDriveNumber() const
   1.998 +	{
   1.999 +	TInt drive = KErrNotFound;
  1.1000 +	
  1.1001 +	// access custom resource drive attribute  
  1.1002 +	if (HAL::Get(HAL::ECustomResourceDrive, drive) == KErrNone)
  1.1003 +		{
  1.1004 +		// check that drive is valid
  1.1005 +		if (drive>=EDriveA && drive<=EDriveZ)
  1.1006 +			return drive;	 
  1.1007 +		}
  1.1008 +		    		    
  1.1009 +	// access system drive attribute  
  1.1010 +	// (Note that ESystemDrive is deprecated. It is checked here 
  1.1011 +	// solely for backward compatibility reasons.)		
  1.1012 +	if (HAL::Get(HAL::ESystemDrive, drive) == KErrNone)
  1.1013 +		{
  1.1014 +		// check that drive is valid
  1.1015 +		if (drive>=EDriveA && drive<=EDriveZ)
  1.1016 +				return drive;
  1.1017 +		}		
  1.1018 + 
  1.1019 +	return KErrNotFound;
  1.1020 +	}
  1.1021 +
  1.1022 +
  1.1023 +
  1.1024 +/** Get the value of the system drive.
  1.1025 +
  1.1026 +The system drive can be set to one of the built-in read/write drives. Which 
  1.1027 +drive is used is hardware-dependent. On some hardware, there may not be a 
  1.1028 +system drive. The system drive is used as the drive on which localisable files 
  1.1029 +are searched for. This enables a phone to be localised dynamically, using 
  1.1030 +files not in the ROM.
  1.1031 +
  1.1032 +@param aDriveNumber On return, contains the drive number of the system drive.
  1.1033 +@return KErrNone is always returned.
  1.1034 +
  1.1035 +@deprecated This method has been replaced by (and now internally calls) 
  1.1036 +RFs:GetSystemDrive, which always returns a valid drive number.
  1.1037 +
  1.1038 +@see BaflUtils::NearestLanguageFile
  1.1039 +@see RFs::GetSystemDrive
  1.1040 +*/
  1.1041 +EXPORT_C TInt BaflUtils::GetSystemDrive(TDriveNumber& aDriveNumber)
  1.1042 +	{
  1.1043 +	aDriveNumber = RFs::GetSystemDrive();
  1.1044 +	return KErrNone;
  1.1045 +	}
  1.1046 +
  1.1047 +
  1.1048 +/** Set most appropriate extension language code for filename and set corresponding language.
  1.1049 +
  1.1050 +Symbian uses numeric values to identify natural languages as specified by the TLanguage enumeration
  1.1051 +defined in e32const.h. These values are used at the end of filename extensions to identify the
  1.1052 +languages pertaining to files which have language specific variants such as resource files.
  1.1053 +For instance filename.r01 and filename.r02 would be the English and French versions of the
  1.1054 +resource file filename.rsc. Language codes can be between 2 to 5 digits in length.
  1.1055 +
  1.1056 +Starting from Symbian OS v7.0 this function constructs and uses a language downgrade path which 
  1.1057 +consists of up to sixteen TLanguage values the first of which is the ideal language followed by 
  1.1058 +the language of the current locale. Up to the next three can be customised using 
  1.1059 +TLocale::SetLanguageDowngrade(). The rest of the language downgrade path is based on a 
  1.1060 +table of language near equivalence which is internal to Symbian.
  1.1061 +
  1.1062 +This function searches the custom resource drive (if set, retrieved from HAL)
  1.1063 +and then searches the optional drive specified in aName or 'Z:' if none is 
  1.1064 +specified in aName. The custom resource drive is retrieved from the HAL 
  1.1065 +attribute ECustomResourceDrive if set, if not set it will retrieve the legacy 
  1.1066 +value set in the legacy HAL attribute ESystemDrive. No custom resource drive 
  1.1067 +is searched if neither are set.  
  1.1068 +Note - setting the custom resource drive will reduce the performance of this 
  1.1069 +routine which will adversely affect device performance e.g. at boot up.
  1.1070 +On NAND Flash based devices use of a composite Z: drive file system made up of multiple 
  1.1071 +ROM images is the preferred mechanism for customising language resources on devices in 
  1.1072 +Symbian OS 9.2 onwards, see Developer Library » Base Porting Guide » Porting: background 
  1.1073 +information » NAND flash » NAND Flash image format. Thus use of the custom resource drive 
  1.1074 +HAL attribute is effectively obsolete.
  1.1075 +  
  1.1076 +The last two characters of aName are removed along with any digits which appear before them.
  1.1077 +Then language codes specified in the constructed language downgrade path are appended in turn to 
  1.1078 +aName as a match is searched for in the file system. In case no match is found using the 
  1.1079 +constructed language downgradepath then files in the specified directory are searched for a 
  1.1080 +suitable extension with preference given to the one specified if present. In cases where a 
  1.1081 +match takes place the aName and aLanguage arguments are updated otherwise aName is left 
  1.1082 +unchanged and aLanguage is set to ELangNone.
  1.1083 +
  1.1084 +Here are some examples of correct and incorrect function usage with different aName inputs, 
  1.1085 +file system state and downgrade paths as follows: 
  1.1086 +
  1.1087 +@code
  1.1088 +Following files exist:
  1.1089 +C:\\abc.rsc  - Language Neutral resource file.
  1.1090 +C:\\abc.r01  - Resource file for the English language.
  1.1091 +C:\\abc.r10  - Resource file for the American-English language.
  1.1092 +C:\\abc.r160 - Resource file for the English as appropriate in Japan.
  1.1093 +
  1.1094 +Constructed Downgrade Language Path cases:
  1.1095 +- Case 1. (ELangAmerican -> ELangEnglish -> ELangNone).
  1.1096 +- Case 2. (ELangEnglish_Japan -> ELangEnglish -> ELangNone).
  1.1097 +- Case 3. Same as case 1, However "C:\\abc.r10" is deleted prior to the function call.
  1.1098 +- Case 4. Same as case 1, However both "C:\\abc.r01" and "C:\\abc.r10" are deleted prior to the function call.
  1.1099 +@endcode
  1.1100 +
  1.1101 +@code
  1.1102 +Input aName . . . . Output aName. . . aLanguage . . . . . Description
  1.1103 +--------------------------------------------------------------------------------------------------------------------
  1.1104 +"C:\\abc.rsc" . . . "C:\\abc.r10" . . ELangAmerican . . . Match on first language (Case 1)
  1.1105 +"C:\\abc.r10" . . . "C:\\abc.r10" . . ELangAmerican . . . Match, However it's not the intended use of 
  1.1106 +. . . . . . . . . . . . . . . . . . . . . . . . . . . . . the function (Case 1)
  1.1107 +"C:\\abc.r" . . . . "C:\\abc.r" . . . ELangNone . . . . . The no. of characters in the suffix is less than 
  1.1108 +. . . . . . . . . . . . . . . . . . . . . . . . . . . . . KInvNameAndMinSuffixLength(2)(Case 1)
  1.1109 +"C:\\abc.". . . . . "C:\\abc.". . . . ELangNone . . . . . Invalid Suffix: The filename ends with a period(Case 1)
  1.1110 +"C:\\abc.r123456" . "C:\\abc.r123456" ELangNone . . . . . Invalid Suffix: The no. of digits in the suffix is greater 
  1.1111 +. . . . . . . . . . . . . . . . . . . . . . . . . . . . . than KMaxSuffixLength(5) (Case 1)
  1.1112 +"C:\\abc.10". . . . "C:\\abc.10 . . . ELangNone . . . . . Invalid Suffix: There's no proceeding alphabetical 
  1.1113 +. . . . . . . . . . . . . . . . . . . . . . . . . . . . . characters in the suffix (Case 1)
  1.1114 +"\\abc.rsc" . . . . "\\abc.rsc" . . . ELangNone . . . . . No drive so Z: search, no match (Case 1)
  1.1115 +"C:\\abc.rsc" . . . "C:\\abc.r160". . ELangEnglish_Japan. Match for language file 3 digits long (Case 2)
  1.1116 +"C:\\abc.rsc" . . . "C:\\abc.r01" . . ELangEnglish. . . . Match on second language (Case 3)
  1.1117 +"C:\\abc.rsc" . . . "C:\\abc.rsc" . . ELangNone . . . . . No corresponding langauge file match found (Case 4)
  1.1118 +---------------------------------------------------------------------------------------------------------------------
  1.1119 +@endcode
  1.1120 +
  1.1121 +@param aFs File server session.
  1.1122 +@param aName Optional drive specification, followed by optional path name,
  1.1123 +followed by basename for filename, followed by period and extension.
  1.1124 +On return, in case of a match, this is replaced by the language-specific version
  1.1125 +which consists of the last two characters of the extension plus any preceding
  1.1126 +numeric characters being replaced by the language code. Remains unchanged when there's no match 
  1.1127 +@param aLanguage On return, in case of a match, this is replaced by the corresponding language.
  1.1128 +In case of no match it's set to ELangNone.
  1.1129 +@see TLanguage
  1.1130 +@see BaflUtils::GetDowngradePathL
  1.1131 +@see TLocale::SetLanguageDowngrade
  1.1132 +*/
  1.1133 +EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage)
  1.1134 +	{
  1.1135 +#if defined(DO_PROFILING)
  1.1136 +	RDebug::ProfileReset(FIRST_PROFILE_INDEX, PROFILE_COUNT);
  1.1137 +	RDebug::ProfileStart(PROFILE_INDEX_1);
  1.1138 +#endif
  1.1139 +	TNearestLanguageFileFinder finder(aFs);
  1.1140 +	TBool goodSuffix=finder.SetFileName(aName);
  1.1141 +	
  1.1142 +	// Only continue if the suffix is good.
  1.1143 +	if(goodSuffix)
  1.1144 +		{
  1.1145 +		// add preset customised resource drive to drive list  
  1.1146 +		// Note that errors returned from AddCustomResourceDrive are ignored. This is because if 
  1.1147 +		// a custom resource drive has not been found we still want to continue on with searching 
  1.1148 +		// other drives according to our algorithm
  1.1149 +		finder.AddCustomResourceDrive();
  1.1150 +		
  1.1151 +		TLocale locale;
  1.1152 +		TLanguage idealLanguage;
  1.1153 +		idealLanguage = IdealLanguage();
  1.1154 +		MakeLanguageDowngradePath(finder.iPath, User::Language(), idealLanguage, locale);
  1.1155 +		if (!finder.FindLanguageAndDrive()
  1.1156 +			&& KErrNone != finder.FindFirstLanguageFileAndDrive())
  1.1157 +			finder.RepairFileName();
  1.1158 +		aLanguage = finder.Language();
  1.1159 +		}
  1.1160 +		
  1.1161 +#if defined(DO_PROFILING)
  1.1162 +	RDebug::ProfileEnd(PROFILE_INDEX_1);
  1.1163 +	TProfile profile[PROFILE_COUNT];
  1.1164 +	RDebug::ProfileResult(&profile[0], FIRST_PROFILE_INDEX, PROFILE_COUNT);
  1.1165 +	if(goodSuffix)
  1.1166 +		{
  1.1167 +		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);
  1.1168 +		}
  1.1169 +	else
  1.1170 +		{
  1.1171 +		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);
  1.1172 +		}
  1.1173 +#endif
  1.1174 +	}
  1.1175 +	
  1.1176 +/** Searches for the file with the correct language extension for the language 
  1.1177 +of the current locale, or failing this, the best matching file.
  1.1178 +
  1.1179 +@param aFs File server session.
  1.1180 +@param aName File name as it would be without a language-specific extension. 
  1.1181 +On return, this is changed to the language-specific version. If no such file 
  1.1182 +is found, the name is unchanged.
  1.1183 +@see BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage) */
  1.1184 +EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName)
  1.1185 +	{
  1.1186 +	TLanguage language;
  1.1187 +	
  1.1188 +	NearestLanguageFile( aFs, aName, language);
  1.1189 +	
  1.1190 +	(void)language;
  1.1191 +	}
  1.1192 +
  1.1193 +/** Set the ideal language for the thread. 
  1.1194 +This interface is intended for the use of UIKON only.
  1.1195 +
  1.1196 +@param aLanguage Ideal language.
  1.1197 +@return KErrNone, if successful; KErrNoMemory if there is not enough memory @see TLanguage
  1.1198 +@see BaflUtils::NearestLanguageFile() 
  1.1199 +@internalAll */
  1.1200 +EXPORT_C TInt BaflUtils::SetIdealLanguage(TLanguage aLanguage)
  1.1201 +	{
  1.1202 +	TLanguage* langPtr=(TLanguage*)Dll::Tls();
  1.1203 +	if( langPtr==NULL)
  1.1204 +		{
  1.1205 +		langPtr=(TLanguage*)User::Alloc(sizeof(TLanguage));
  1.1206 +		
  1.1207 +		if(!langPtr) 
  1.1208 +			return(KErrNoMemory);
  1.1209 +		
  1.1210 +		TInt ret=Dll::SetTls(langPtr);
  1.1211 +		
  1.1212 +		if(ret!=KErrNone)
  1.1213 +			return(ret);
  1.1214 +		}
  1.1215 +	*langPtr=aLanguage;
  1.1216 +	return(KErrNone);
  1.1217 +	}
  1.1218 +	
  1.1219 +/** Get the ideal language of the thread. 
  1.1220 +This interface is intended for the use of UIKON only.
  1.1221 +
  1.1222 +@return Ideal language if set, ELangNone if not set
  1.1223 +@see BaflUtils::NearestLanguageFile() 
  1.1224 +@internalAll */
  1.1225 +EXPORT_C TLanguage BaflUtils::IdealLanguage()
  1.1226 +	{
  1.1227 +	TLanguage* langPtr=(TLanguage*)Dll::Tls();
  1.1228 +	
  1.1229 +	if( langPtr==NULL)
  1.1230 +		{
  1.1231 +		return(ELangNone);
  1.1232 +		}
  1.1233 +
  1.1234 +	return(*langPtr);
  1.1235 +	}
  1.1236 +	
  1.1237 +EXPORT_C void BaflUtils::ReleaseIdealLanguage()
  1.1238 +/** Releases the ideal language store if it has been allocated. 
  1.1239 +This interface is intended for the use of UIKON only.
  1.1240 +
  1.1241 +@internalAll */
  1.1242 +	{
  1.1243 +	TLanguage* aLanguage=(TLanguage*)Dll::Tls();
  1.1244 +	if( aLanguage==NULL)
  1.1245 +		return;
  1.1246 +	
  1.1247 +	delete aLanguage;
  1.1248 +	Dll::FreeTls();
  1.1249 +	}
  1.1250 +
  1.1251 +EXPORT_C TInt BaflUtils::IsFolder(const RFs& aFs, const TDesC& aFullName, TBool& aIsFolder)
  1.1252 +/** Checks if the specified item is a folder.
  1.1253 +
  1.1254 +@param aFs File server session
  1.1255 +@param aFullName Name to check
  1.1256 +@param aIsFolder ETrue if aFullName is a folder, otherwise EFalse
  1.1257 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1258 +codes */
  1.1259 +	{
  1.1260 +	TParsePtrC parse(aFullName);
  1.1261 +	if ((parse.DriveAndPath().Length() == KDriveAndPathLength) && (aFullName.Length() == KDriveAndPathLength))	
  1.1262 +		{
  1.1263 +		aIsFolder = ETrue;
  1.1264 +		return(KErrNone);
  1.1265 +		}
  1.1266 +	TEntry entry;
  1.1267 +	TInt retcode = aFs.Entry(aFullName, entry);
  1.1268 +	if (retcode == KErrNone)
  1.1269 +		aIsFolder = ((entry.iAtt & KEntryAttDir)==KEntryAttDir);
  1.1270 +
  1.1271 +	return(retcode);
  1.1272 +	}
  1.1273 +
  1.1274 +
  1.1275 +EXPORT_C TBool BaflUtils::FolderExists(RFs& aFs, const TDesC& aFolderName)
  1.1276 +/** Tests whether a folder exists.
  1.1277 +
  1.1278 +The folder is specified in a path. The path can omit the drive letter, in 
  1.1279 +which case the drive letter is taken from the session path.
  1.1280 +
  1.1281 +If the path is badly formed, for instance if it contains illegal characters, 
  1.1282 +or any directory name consists of a single or double dot, or any directory 
  1.1283 +name includes wildcard characters, the function returns EFalse.
  1.1284 +
  1.1285 +If a filename is included in the path, it is ignored (the existence 
  1.1286 +of the file will not be checked).  However if included, it must not
  1.1287 +be badly formed - this will cause the function to return EFalse.  
  1.1288 +If no filename is specified, the path should end in a backslash.
  1.1289 +
  1.1290 +Examples of valid paths (returning ETrue):
  1.1291 +C:\; \; C:\Readme.txt; C:\system\data\; \system\data\Anyfile.dat
  1.1292 +
  1.1293 +Examples of invalid paths (returning EFalse):
  1.1294 +C:\FolderDoesntExist\; ..\system\; C:\Wild*card\; C:\system\data\Bad>File.txt
  1.1295 +
  1.1296 +@param aFs A connected session with the file server.
  1.1297 +@param aFolderName A path specifying the folder to test for.
  1.1298 +@return ETrue if the folder specified in aFolderName exists, EFalse if not. 
  1.1299 +EFalse is also returned if the specified path is badly formed. */
  1.1300 +	{	
  1.1301 +	if (aFolderName.Length()==0)
  1.1302 +		{return EFalse;}
  1.1303 +	
  1.1304 +	TParse parse;
  1.1305 +		 	
  1.1306 + 	TInt retcode = parse.SetNoWild(aFolderName, NULL, NULL);
  1.1307 + 	
  1.1308 + 	if (retcode != KErrNone)
  1.1309 + 		{return EFalse;}
  1.1310 + 			
  1.1311 + 	if (parse.NameOrExtPresent())
  1.1312 + 		if (!aFs.IsValidName(aFolderName))
  1.1313 + 			{return EFalse;}
  1.1314 + 	
  1.1315 + 	TPtrC dirName = parse.DriveAndPath();
  1.1316 + 	RDir dir;
  1.1317 +    retcode = dir.Open(aFs,dirName,0);
  1.1318 + 	if (retcode == KErrNone)
  1.1319 + 		{dir.Close();}
  1.1320 + 	return (retcode == KErrNone);
  1.1321 +	}
  1.1322 +
  1.1323 +
  1.1324 +EXPORT_C TFileName BaflUtils::FolderNameFromFullName(const TDesC& aFullName) 
  1.1325 +/** Gets the folder name from a path.
  1.1326 +
  1.1327 +A valid full name should have a drive associated with it
  1.1328 +e.g ("a:\\" - "z:\\")("a:" - "z:")("c:\\system\data\file.txt")
  1.1329 +Invalid entry will have no drive and cause a panic EBafPanicBadOpenArg
  1.1330 +For example, if the path is "c:\documents\word\mydoc1", then "word" is returned.
  1.1331 +							"c:"						then "c:" is returned
  1.1332 +							"c:\\"						then "c:\" is returned
  1.1333 +							"c:\\mydoc1.txt				then "c:\" is returned
  1.1334 +
  1.1335 +@param aFullName A path.
  1.1336 +@return The folder name. */
  1.1337 +	{
  1.1338 +	TParsePtrC parse(aFullName);
  1.1339 +	
  1.1340 +	__ASSERT_ALWAYS(parse.DrivePresent(),Panic(EBafPanicBadOpenArg));
  1.1341 +	
  1.1342 +	TFileName folderName = parse.Path();
  1.1343 +	//If the path name has no associated path(e.g "c:") or path='\'(e.g "c:\\", "c:\\file.txt")
  1.1344 +	//then the folder name is just equal to drivename
  1.1345 +	
  1.1346 +	TBuf<1> pathSeparator;
  1.1347 +	pathSeparator.Append(KPathDelimiter);
  1.1348 +
  1.1349 +	if (folderName.Length()==0 || folderName==pathSeparator)
  1.1350 +		return (parse.DriveAndPath());
  1.1351 +	//else just get the foldername
  1.1352 +	TInt len = folderName.Length();
  1.1353 +	TInt pos = --len;
  1.1354 +	while (--pos)
  1.1355 +		if (folderName.Mid(pos, 1)==pathSeparator)
  1.1356 +			break;
  1.1357 +	folderName.Delete(len, 1);
  1.1358 +	folderName.Delete(0, pos+1);
  1.1359 +	return(folderName);
  1.1360 +	}
  1.1361 +
  1.1362 +
  1.1363 +EXPORT_C TFileName BaflUtils::DriveAndPathFromFullName(const TDesC& aFullName) 
  1.1364 +/** Gets the drive letter and path from a file name. 
  1.1365 +
  1.1366 +This is in the form: drive-letter:\\path\\. The drive letter is folded using 
  1.1367 +class TCharF. 
  1.1368 +
  1.1369 +@param aFullName File name
  1.1370 +@return The drive and path */
  1.1371 +	{
  1.1372 +	TParsePtrC parse(aFullName);
  1.1373 +	return (parse.DriveAndPath());
  1.1374 +	}
  1.1375 +
  1.1376 +
  1.1377 +EXPORT_C TFileName BaflUtils::RootFolderPath(const TBuf<1> aDriveLetter)
  1.1378 +/** Gets the root folder for the specified drive.
  1.1379 +
  1.1380 +If aDriveLetter is an alphabet(lowercase or uppercase) then it will return
  1.1381 +the TFileName which is simply the drive letter plus ":\"
  1.1382 +If this is not the case, the function will panic with panic code EBafPanicBadOpenArg
  1.1383 +
  1.1384 +@param aDriveLetter Drive letter
  1.1385 +@return Root folder */
  1.1386 +	{
  1.1387 +	TChar driveLetter(aDriveLetter[0]);
  1.1388 +	driveLetter.LowerCase();
  1.1389 +	TInt aDriveNumber=driveLetter-TChar('a');
  1.1390 +	__ASSERT_ALWAYS(aDriveNumber>= EDriveA && aDriveNumber <= EDriveZ,Panic(EBafPanicBadOpenArg));
  1.1391 +	
  1.1392 +	TFileName rootFolderPath = aDriveLetter;
  1.1393 +	rootFolderPath.Append(_L(":\\"));
  1.1394 +	return rootFolderPath;
  1.1395 +	}
  1.1396 +
  1.1397 +
  1.1398 +EXPORT_C void BaflUtils::AbbreviateFileName(const TFileName& aOriginalFileName, TDes& aAbbreviatedFileName)
  1.1399 +/** Abbreviates a file name.
  1.1400 +
  1.1401 +If aOriginalFileName is less than the maximum length of aAbbreviatedFileName, 
  1.1402 +then the name is simply copied to aAbbreviatedFileName.
  1.1403 +
  1.1404 +If this is not so, then the left-most characters of aOriginalFileName are 
  1.1405 +copied to aAbbreviatedFileName, up to aAbbreviatedFileName's maximum length-1. 
  1.1406 +aAbbreviatedFileName's first character is set to be an ellipsis.
  1.1407 +
  1.1408 +@param aOriginalFileName Original file name
  1.1409 +@param aAbbreviatedFileName On return, abbreviated file name */
  1.1410 +	{
  1.1411 +	TInt maxWidthInChars = aAbbreviatedFileName.MaxLength();
  1.1412 +	if (aOriginalFileName.Length() <= maxWidthInChars)
  1.1413 +		{
  1.1414 +		aAbbreviatedFileName = aOriginalFileName;
  1.1415 +		return;
  1.1416 +		}
  1.1417 +	TChar ellipsis(0x2026);
  1.1418 +	--maxWidthInChars;  // since the ellipsis will be the first char in aAbbreviatedFileName
  1.1419 +	aAbbreviatedFileName.Zero();
  1.1420 +	aAbbreviatedFileName.Append(ellipsis);
  1.1421 +	aAbbreviatedFileName.Append(aOriginalFileName.Mid((aOriginalFileName.Length() - 1) - maxWidthInChars + 1, maxWidthInChars));
  1.1422 +	}
  1.1423 +
  1.1424 +
  1.1425 +EXPORT_C TBool BaflUtils::UidTypeMatches(const TUidType& aFileUid, const TUidType& aMatchUid)
  1.1426 +/** Tests whether two UID types match.
  1.1427 +
  1.1428 +A match is made if each UID in aMatchUid is either identical to the corresponding 
  1.1429 +one in aFileUid, or is KNullUid.
  1.1430 +
  1.1431 +@param aFileUid The UID type to match
  1.1432 +@param aMatchUid The UID type to match against
  1.1433 +@return ETrue if the UIDs match, otherwise EFalse */
  1.1434 +	{
  1.1435 +	for (TInt i=0; i<KMaxCheckedUid; i++)
  1.1436 +		{
  1.1437 +		if (aMatchUid[i] == KNullUid)
  1.1438 +			continue;
  1.1439 +		if (aMatchUid[i] != aFileUid[i])
  1.1440 +			return(EFalse);
  1.1441 +		}
  1.1442 +	return(ETrue);
  1.1443 +	}
  1.1444 +
  1.1445 +
  1.1446 +EXPORT_C TInt BaflUtils::Parse(const TDesC& aName)
  1.1447 +/** Checks if a specified file name can be parsed.
  1.1448 +
  1.1449 +@param aName Name to parse
  1.1450 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1451 +codes */
  1.1452 +	{ // keeps a TParse on the stack for the minimum time possible
  1.1453 +	TParse parse;
  1.1454 +	return parse.Set(aName,NULL,NULL);
  1.1455 +	}
  1.1456 +
  1.1457 +
  1.1458 +EXPORT_C TInt BaflUtils::ValidateFolderNameTypedByUserL(const RFs& aFs, const TDesC& aFolderNameTypedByUser, const TDesC& aCurrentPath, TFileName& aNewFolderFullName)
  1.1459 +/** Checks if a folder name (without drive or path) is valid and returns the full 
  1.1460 +name of the folder.
  1.1461 +
  1.1462 +@param aFs File server session
  1.1463 +@param aFolderNameTypedByUser Folder name to check
  1.1464 +@param aCurrentPath Path to which to add the folder
  1.1465 +@param aNewFolderFullName aFolderNameTypedByUser appended to aCurrentPath
  1.1466 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1467 +codes */
  1.1468 +	{
  1.1469 +	if (aFolderNameTypedByUser.Length() == 0)
  1.1470 +		return KErrArgument;	// R_EIK_TBUF_NO_FOLDERNAME_SPECIFIED;
  1.1471 +
  1.1472 +	TParse* targetParse = new(ELeave) TParse;
  1.1473 +	CleanupStack::PushL(targetParse);
  1.1474 +
  1.1475 +	TInt retcode = targetParse->Set(aFolderNameTypedByUser, NULL, NULL);
  1.1476 +	User::LeaveIfError(retcode);
  1.1477 +	if (targetParse->DrivePresent() || targetParse->PathPresent())
  1.1478 +		{
  1.1479 +		 CleanupStack::PopAndDestroy();
  1.1480 +		 return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
  1.1481 +		}
  1.1482 +
  1.1483 +	if (!(aFs.IsValidName(aFolderNameTypedByUser)))
  1.1484 +		{
  1.1485 +		 CleanupStack::PopAndDestroy();
  1.1486 +		 return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
  1.1487 +		}
  1.1488 +		 
  1.1489 +
  1.1490 +	if ((aCurrentPath.Length() + aFolderNameTypedByUser.Length() + 1) > KMaxFileName)
  1.1491 +		{
  1.1492 +		 CleanupStack::PopAndDestroy();
  1.1493 +		 return KErrTooBig;		//R_EIK_TBUF_FOLDERNAME_TOO_LONG;
  1.1494 +		}
  1.1495 +		 
  1.1496 +
  1.1497 +	//TFileName newFolderFullName = aCurrentPath;
  1.1498 +	aNewFolderFullName = aCurrentPath;
  1.1499 +	if ((aNewFolderFullName.Length() + aFolderNameTypedByUser.Length() + 1) <= KMaxFileName)
  1.1500 +		{
  1.1501 +		aNewFolderFullName.Append(aFolderNameTypedByUser);
  1.1502 +		aNewFolderFullName.Append(KPathDelimiter);
  1.1503 +		}
  1.1504 +	else
  1.1505 +		{
  1.1506 +		CleanupStack::PopAndDestroy();
  1.1507 +		return KErrOverflow;
  1.1508 +		}
  1.1509 +
  1.1510 +	retcode = targetParse->Set(aNewFolderFullName, NULL, NULL);
  1.1511 +	if (retcode != KErrNone)
  1.1512 +		{
  1.1513 +		CleanupStack::PopAndDestroy();
  1.1514 +		return KErrBadName;	// R_EIK_TBUF_INVALID_FOLDER_NAME;
  1.1515 +		}
  1.1516 +	
  1.1517 +	CleanupStack::PopAndDestroy();
  1.1518 +
  1.1519 +	return(KErrNone);
  1.1520 +	}
  1.1521 +
  1.1522 +void BaflUtils::DoCopyFileL(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch)
  1.1523 +	{
  1.1524 +	CFileMan* fileMan=CFileMan::NewL(aFs);
  1.1525 +	CleanupStack::PushL(fileMan);
  1.1526 +	User::LeaveIfError(fileMan->Copy(aSourceFullName,aTargetFullName,aSwitch));
  1.1527 +	CleanupStack::PopAndDestroy(); // fileMan
  1.1528 +	}
  1.1529 +
  1.1530 +
  1.1531 +EXPORT_C TInt BaflUtils::CopyFile(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch)
  1.1532 +/** Copies one or more files.
  1.1533 +
  1.1534 +For more details, 
  1.1535 +@see CFileMan::Copy()
  1.1536 +@since     5.1
  1.1537 +@param     aFs File server session
  1.1538 +@param     aSourceFullName Path indicating the file(s) to be copied. Any path
  1.1539 +components that are not specified here will be taken from the session path.
  1.1540 +@param     aTargetFullName Path indicating the directory into which the file(s)
  1.1541 +are to be copied
  1.1542 +@param     aSwitch=CFileMan::EOverWrite Set this to zero for no overwriting and 
  1.1543 +no recursion; CFileMan::EOverWrite to overwrite files with the same name; or 
  1.1544 +CFileMan::ERecurse for recursion.
  1.1545 +@return   KErrNone if successful, otherwise another of the system-wide error 
  1.1546 +codes.*/
  1.1547 +	{
  1.1548 +	TRAPD(err,DoCopyFileL(aFs,aSourceFullName,aTargetFullName,aSwitch));
  1.1549 +	return err;
  1.1550 +	}
  1.1551 +
  1.1552 +void BaflUtils::DoDeleteFileL(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch)
  1.1553 +	{
  1.1554 +	CFileMan* fileMan=CFileMan::NewL(aFs);
  1.1555 +	CleanupStack::PushL(fileMan);
  1.1556 +	User::LeaveIfError(fileMan->Delete(aSourceFullName,aSwitch));
  1.1557 +	CleanupStack::PopAndDestroy(); // fileMan
  1.1558 +	}
  1.1559 +
  1.1560 +
  1.1561 +EXPORT_C TInt BaflUtils::DeleteFile(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch)
  1.1562 +/** Deletes one or more files.
  1.1563 +
  1.1564 +For more details,
  1.1565 +@see CFileMan::Delete().
  1.1566 +@since 5.1
  1.1567 +@param aFs File server session
  1.1568 +@param aSourceFullName Path indicating the file(s) to be deleted. May either
  1.1569 +be a full path, or relative to the session path. Use wildcards to specify 
  1.1570 +more than one file.
  1.1571 +@param aSwitch=0  Specify CFileMan::ERecurse for recursion,
  1.1572 +zero for no recursion.
  1.1573 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1574 +codes. */	
  1.1575 +    {
  1.1576 +	TRAPD(err,DoDeleteFileL(aFs,aSourceFullName,aSwitch));
  1.1577 +	return err;
  1.1578 +	}
  1.1579 +
  1.1580 +void BaflUtils::DoRenameFileL(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch)
  1.1581 +	{
  1.1582 +	CFileMan* fileMan=CFileMan::NewL(aFs);
  1.1583 +	CleanupStack::PushL(fileMan);
  1.1584 +	User::LeaveIfError(fileMan->Rename(aOldFullName,aNewFullName,aSwitch));
  1.1585 +	CleanupStack::PopAndDestroy(); // fileMan
  1.1586 +	}
  1.1587 +
  1.1588 +
  1.1589 +EXPORT_C TInt BaflUtils::RenameFile(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch)
  1.1590 +/**  Renames or moves one or more files or directories.
  1.1591 +
  1.1592 +It can be used to move one or more files by specifying different
  1.1593 +destination and source directories.
  1.1594 +For more details, 
  1.1595 +@see CFileMan::Rename().
  1.1596 +@since 5.1
  1.1597 +@param aFs File server session
  1.1598 +@param aOldFullName Path specifying the file(s) to be renamed.
  1.1599 +@param aNewFullName Path specifying the new name for the files and/or the
  1.1600 +new directory. Any directories specified in this path that do not exist will 
  1.1601 +be created.
  1.1602 +@param aSwitch=CFileMan::EOverWrite  Specify zero for no overwriting, or
  1.1603 +CFileMan::EOverWrite to overwrite files with the same name. This 
  1.1604 +function cannot operate recursively.
  1.1605 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1606 +codes. */	
  1.1607 +    {
  1.1608 +	TRAPD(err,DoRenameFileL(aFs,aOldFullName,aNewFullName,aSwitch));
  1.1609 +	return err;
  1.1610 +	}
  1.1611 +
  1.1612 +
  1.1613 +EXPORT_C TInt BaflUtils::CheckWhetherFullNameRefersToFolder(const TDesC& aFullName, TBool& aIsFolder)
  1.1614 +/** Checks if a string is a valid folder name.
  1.1615 +
  1.1616 +@param aFullName String to check
  1.1617 +@param aIsFolder ETrue if aFullName is a valid folder name, otherwise EFalse
  1.1618 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1619 +codes (probably because aFullName cannot be parsed). */
  1.1620 +	{
  1.1621 +	aIsFolder = EFalse;
  1.1622 +	TInt retcode = BaflUtils::Parse(aFullName);
  1.1623 +	if (retcode != KErrNone)
  1.1624 +		return(retcode);
  1.1625 +	TParsePtrC parse(aFullName);
  1.1626 +	if (! parse.NameOrExtPresent())
  1.1627 +		aIsFolder = ETrue;
  1.1628 +	return(KErrNone);
  1.1629 +	}
  1.1630 +
  1.1631 +EXPORT_C TInt BaflUtils::MostSignificantPartOfFullName(const TDesC& aFullName, TFileName& aMostSignificantPart)
  1.1632 +/** Gets the folder name if the specified item is a valid folder name, otherwise 
  1.1633 +gets the file name.
  1.1634 +
  1.1635 +@param aFullName Item to parse
  1.1636 +@param aMostSignificantPart Folder or file name
  1.1637 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1638 +codes */
  1.1639 +	{
  1.1640 +	TBool entryIsAFolder;
  1.1641 +	TInt retcode = CheckWhetherFullNameRefersToFolder(aFullName, entryIsAFolder);
  1.1642 +	if (retcode != KErrNone)
  1.1643 +		return(retcode);
  1.1644 +	if (entryIsAFolder)
  1.1645 +		{
  1.1646 +		aMostSignificantPart = FolderNameFromFullName(aFullName);
  1.1647 +		return (KErrNone);
  1.1648 +		}
  1.1649 +	// assume aFullName refers to a file
  1.1650 +	TParsePtrC parse(aFullName);
  1.1651 +	aMostSignificantPart = parse.NameAndExt();
  1.1652 +	return(KErrNone);
  1.1653 +	}
  1.1654 +
  1.1655 +EXPORT_C TInt BaflUtils::CheckFolder(RFs& aFs, const TDesC& aFolderName)
  1.1656 +/** Checks that the specified folder can be opened.
  1.1657 +
  1.1658 +@param aFs File server session
  1.1659 +@param aFolderName Folder to check
  1.1660 +@return KErrNone if successful, otherwise another of the system-wide error 
  1.1661 +codes */
  1.1662 +	{
  1.1663 +    RDir dir;
  1.1664 +    TInt retcode = dir.Open(aFs, aFolderName, 0);
  1.1665 +	if (retcode == KErrNone)
  1.1666 +		dir.Close();
  1.1667 +	return (retcode);
  1.1668 +	}
  1.1669 +
  1.1670 +/**
  1.1671 +Checks if the specified drive is read-only.
  1.1672 +Checks that the KMediaAttWriteProtected and EMediaRom flags are both set.
  1.1673 +
  1.1674 +@param aFs File server session
  1.1675 +@param aFullName File name, including drive
  1.1676 +@param aIsReadOnly On return, ETrue if the drive is read-only, otherwise EFalse
  1.1677 +@return KErrNone if successful, otherwise another of the system-wide error codes
  1.1678 +@see BaflUtils::DriveIsReadOnlyInternal
  1.1679 +*/
  1.1680 +EXPORT_C TInt BaflUtils::DiskIsReadOnly(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnly)
  1.1681 +	{
  1.1682 +	TInt retcode=BaflUtils::Parse(aFullName);
  1.1683 +	if (retcode!=KErrNone)
  1.1684 +		return retcode;
  1.1685 +	TParsePtrC parse(aFullName);
  1.1686 +	if (!parse.DrivePresent())
  1.1687 +		return KErrBadName;
  1.1688 +	TBuf<1> drive=parse.Drive().Left(1);
  1.1689 +	TChar driveLetter=drive[0];
  1.1690 +	TInt driveId=0;
  1.1691 +	retcode=RFs::CharToDrive(driveLetter,driveId);
  1.1692 +	if (retcode!=KErrNone)
  1.1693 +		return retcode;
  1.1694 +	TVolumeInfo volInfo;
  1.1695 +	retcode=aFs.Volume(volInfo,driveId);
  1.1696 +	if (retcode!=KErrNone)
  1.1697 +		return retcode;
  1.1698 +	aIsReadOnly=(volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected || volInfo.iDrive.iType==EMediaRom);
  1.1699 +	return KErrNone;
  1.1700 +	}
  1.1701 +
  1.1702 +/** 
  1.1703 +Checks if the specified drive is read-only and is an internal drive i.e. non-removable.
  1.1704 +Checks that the KMediaAttWriteProtected and KDriveAttInternal flags are both set.
  1.1705 +
  1.1706 +@param aFs File server session
  1.1707 +@param aFullName File name, including drive
  1.1708 +@param aIsReadOnlyInternal On return, ETrue if the drive is read-only and internal, otherwise EFalse
  1.1709 +@return KErrNone if successful, otherwise another of the system-wide errors codes 
  1.1710 +*/
  1.1711 +EXPORT_C TInt BaflUtils::DriveIsReadOnlyInternal(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnlyInternal)
  1.1712 +	{
  1.1713 +	TInt retcode=BaflUtils::Parse(aFullName);
  1.1714 +	if (retcode!=KErrNone)
  1.1715 +		return retcode;
  1.1716 +	TParsePtrC parse(aFullName);
  1.1717 +	if (!parse.DrivePresent())
  1.1718 +		return KErrBadName;
  1.1719 +	TBuf<1> drive=parse.Drive().Left(1);
  1.1720 +	TChar driveLetter=drive[0];
  1.1721 +	TInt driveId=0;
  1.1722 +	retcode=RFs::CharToDrive(driveLetter,driveId);
  1.1723 +	if (retcode!=KErrNone)
  1.1724 +		return retcode;
  1.1725 +	TVolumeInfo volInfo;
  1.1726 +	retcode=aFs.Volume(volInfo,driveId);
  1.1727 +	if (retcode!=KErrNone)
  1.1728 +		return retcode;
  1.1729 +	aIsReadOnlyInternal=((volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected)&&(volInfo.iDrive.iDriveAtt&KDriveAttInternal));
  1.1730 +	return KErrNone;
  1.1731 +	}
  1.1732 +
  1.1733 +EXPORT_C void BaflUtils::GetDiskListL(const RFs& aFs,CDesCArray& aArray)
  1.1734 +/** Retrieves a list of all drives on the system.
  1.1735 +
  1.1736 +The file server is interrogated for a list of the drive letters for all available 
  1.1737 +drives. 
  1.1738 +
  1.1739 +On emulator:
  1.1740 +The removable media is represented by drive X: .
  1.1741 +
  1.1742 +On hardware:
  1.1743 +The removable media is represented by drives D: E: F: and G: .
  1.1744 +
  1.1745 +@param aFs A connected session with the file server.
  1.1746 +@param aArray On return, contains the drive letters that correspond to the 
  1.1747 +available drives. The drive letters are uppercase and are in alphabetical 
  1.1748 +order. */
  1.1749 +	{ // static
  1.1750 +	aArray.Reset();
  1.1751 +	TDriveList driveList;
  1.1752 +	User::LeaveIfError(aFs.DriveList(driveList));
  1.1753 +	for (TInt ii=0;ii<KMaxDrives;ii++)
  1.1754 +		{
  1.1755 +		if (driveList[ii])
  1.1756 +			{
  1.1757 +			TChar drive;
  1.1758 +			User::LeaveIfError(aFs.DriveToChar(ii,drive));
  1.1759 +			drive.UpperCase();
  1.1760 +			TBuf<1> buf;
  1.1761 +			buf.Append(drive);
  1.1762 +			aArray.AppendL(buf);
  1.1763 +			}
  1.1764 +		}
  1.1765 +	}
  1.1766 +
  1.1767 +EXPORT_C void BaflUtils::UpdateDiskListL(const RFs& aFs,CDesCArray& aArray,TBool aIncludeRom,TDriveNumber aDriveNumber)
  1.1768 +/** Retrieves a list of all drives present on the system.
  1.1769 +
  1.1770 +The file server is interrogated for a list of the drive letters for all available 
  1.1771 +drives. The drive letter that corresponds to aDriveNumber is added to the 
  1.1772 +list regardless of whether it is present, or is corrupt. Also, the C: drive 
  1.1773 +is forced onto the list, even if corrupt or not present.
  1.1774 +
  1.1775 +On emulator:
  1.1776 +The removable media is represented by drive X: and is forced onto the list 
  1.1777 +unless removed (F5,F4).
  1.1778 +
  1.1779 +On hardware:
  1.1780 +The removable media is represented by drives D: E: F: and G: and is forced 
  1.1781 +onto the list regardless of whether it is present, or is corrupt.
  1.1782 +
  1.1783 +@param aFs A connected session with the file server.
  1.1784 +@param aArray On return, contains the drive letters that correspond to the 
  1.1785 +available drives. The drive letters are uppercase and are in alphabetical 
  1.1786 +order.
  1.1787 +@param aIncludeRom Specify ETrue if the ROM drive should be included in the 
  1.1788 +list, EFalse if not.
  1.1789 +@param aDriveNumber The drive to force into the list, e.g. the drive in the 
  1.1790 +default path. */
  1.1791 +	{ // static
  1.1792 +	aArray.Reset();
  1.1793 +	TDriveList driveList;
  1.1794 +	User::LeaveIfError(aFs.DriveList(driveList));
  1.1795 +	for (TInt ii=0;ii<KMaxDrives;ii++)
  1.1796 +		{
  1.1797 +		if (driveList[ii] || ii==aDriveNumber)
  1.1798 +			{
  1.1799 +			TVolumeInfo vInfo;
  1.1800 +			const TInt err=aFs.Volume(vInfo,TDriveUnit(ii));
  1.1801 +			if (err==KErrNone || err==KErrCorrupt || ii==aDriveNumber || BaflUtils::IsFirstDriveForSocket(TDriveUnit(ii)))
  1.1802 +				{
  1.1803 +				if (ii==EDriveZ && vInfo.iDrive.iDriveAtt&KDriveAttRom && !aIncludeRom)
  1.1804 +					continue;
  1.1805 +				TChar drive;
  1.1806 +				User::LeaveIfError(aFs.DriveToChar(ii,drive));
  1.1807 +				drive.UpperCase();
  1.1808 +				TBuf<1> buf;
  1.1809 +				buf.Append(drive);
  1.1810 +				aArray.AppendL(buf);
  1.1811 +				}
  1.1812 +			}
  1.1813 +		}
  1.1814 +	}
  1.1815 +
  1.1816 +EXPORT_C TBool BaflUtils::IsFirstDriveForSocket(TDriveUnit aDriveUnit)
  1.1817 +/** Tests whether the specified drive corresponds to the primary partition in a 
  1.1818 +removable media slot.
  1.1819 +
  1.1820 +The function assumes that the D: drive corresponds to the primary partition 
  1.1821 +on socket 0, and that the E: drive corresponds to the primary partition on 
  1.1822 +socket 1 (a socket is a slot for removable media). This mapping may not always 
  1.1823 +be the case because it is set up in the variant layer of the Symbian OS.
  1.1824 +
  1.1825 +This function assumes that the drive mappings are contiguous, starting 
  1.1826 +from drive D: .
  1.1827 +
  1.1828 +On emulator:
  1.1829 +The removable media is represented by drive X: only.
  1.1830 +
  1.1831 +@param aDriveUnit The drive to check.
  1.1832 +@return ETrue if the drive is the primary partition in a removable media slot. 
  1.1833 +ETrue is also returned if the drive is A, B or C:. EFalse is returned otherwise. */
  1.1834 +	{ // static
  1.1835 +	TDriveInfoV1Buf buf;
  1.1836 +	UserHal::DriveInfo(buf);
  1.1837 +
  1.1838 +#ifdef __EPOC32__
  1.1839 +	return ((aDriveUnit-EDriveC)<=buf().iTotalSockets);
  1.1840 +#else // emulator
  1.1841 +    return (aDriveUnit==EDriveX || (aDriveUnit-EDriveC)<=buf().iTotalSockets);
  1.1842 +#endif
  1.1843 +	}
  1.1844 +
  1.1845 +EXPORT_C void BaflUtils::RemoveSystemDirectory(CDir& aDir)
  1.1846 +/** Removes "System" from a list of directory entries. 
  1.1847 +
  1.1848 +@param aDir Array of directory entries. */
  1.1849 +	{ // static
  1.1850 +	STATIC_CAST(BaflDir&,aDir).RemoveSystem();
  1.1851 +	}
  1.1852 +
  1.1853 +EXPORT_C TInt BaflUtils::SortByTable(CDir& aDir,CBaflFileSortTable* aTable)
  1.1854 +/** Sorts files by UID.
  1.1855 +
  1.1856 +The caller supplies a table which specifies the order in which files are to 
  1.1857 +be sorted. The files whose UID3 is the first UID in the table appear first. 
  1.1858 +The files whose UID3 is the UID specified second appear next, and so on. Files 
  1.1859 +whose UID3 is not specified in the table, and directories, appear at the end 
  1.1860 +of the list, with directories preceding the files, and with files sorted in 
  1.1861 +ascending order of UID3.
  1.1862 +
  1.1863 +This function is used for customising how lists of application files are sorted.
  1.1864 +
  1.1865 +@param aDir The array of files and directories to sort.
  1.1866 +@param aTable A sort order table containing the UIDs to use in the sort.
  1.1867 +@return KErrNone if successful, otherwise one of the standard error codes. */
  1.1868 +	{ // static
  1.1869 +	return STATIC_CAST(BaflDir&,aDir).SortByTable(aTable);
  1.1870 +	}
  1.1871 +
  1.1872 +EXPORT_C void BaflUtils::GetDowngradePathL(const RFs& aFs, const TLanguage aCurrentLanguage, RArray<TLanguage>& aLanguageArray)
  1.1873 +/** Gets the full language downgrade path for a particular locale.
  1.1874 +
  1.1875 +@param aFs A connected session with the file server.
  1.1876 +@param aCurrentLanguage The language of the locale for which the language downgrade 
  1.1877 +path is required. This language will always be returned as the first language 
  1.1878 +in aLanguageArray. To get the downgrade path for the language of the current 
  1.1879 +locale, specify the language returned by User::Language(). 
  1.1880 +@param aLanguageArray On return, contains the language downgrade path.
  1.1881 +@see BaflUtils::NearestLanguageFile() */
  1.1882 + 	{
  1.1883 + 	TLocale currentLocale; 
  1.1884 +  	TNearestLanguageFileFinder languageDowngradePath(aFs);
  1.1885 +  	TLanguage idealLanguage=IdealLanguage();
  1.1886 +  	MakeLanguageDowngradePath(languageDowngradePath.iPath,aCurrentLanguage,idealLanguage, currentLocale);
  1.1887 + 	aLanguageArray.Reset();
  1.1888 +  	const TLanguage* p=languageDowngradePath.iPath;
  1.1889 +  	while (*p != ELangNone)
  1.1890 +  		{
  1.1891 +		User::LeaveIfError(aLanguageArray.Append(*p));
  1.1892 +  		++p;
  1.1893 +  		}
  1.1894 + 	}
  1.1895 +
  1.1896 +EXPORT_C void BaflUtils::PersistLocale()
  1.1897 +/** Saves the locale settings in TLocale and the currency symbol to file. 
  1.1898 +@deprecated 9.1
  1.1899 +Persistence and initialisation of system locale data will be performed 
  1.1900 +transparently by a separate executable (InilialiseLocale.exe) wich should 
  1.1901 +be loaded as part of the system start procedure.
  1.1902 +*/
  1.1903 +	{
  1.1904 +// Replaced by new repository based locale initialisation mechanism.
  1.1905 +	}
  1.1906 +
  1.1907 +
  1.1908 +
  1.1909 +EXPORT_C TInt BaflUtils::PersistHAL()
  1.1910 +/** Saves the HAL settings to file. 
  1.1911 +This will start a new executable and saves HAL attributes to a file, 
  1.1912 +little delay because of the creation of new process
  1.1913 +@return KErrNone if suceessful, otheriwse system wide error code.
  1.1914 +*/
  1.1915 +	{
  1.1916 +	RProcess process;
  1.1917 +	_LIT(KHALSettings, "HALSettings.exe");
  1.1918 +	_LIT(KCommandLine, "PERSIST");
  1.1919 +	TInt result = process.Create(KHALSettings, KCommandLine);
  1.1920 +	if(result != KErrNone )
  1.1921 +		return result;
  1.1922 +	TRequestStatus status;
  1.1923 +	process.Logon(status);
  1.1924 +	if ( status != KRequestPending)
  1.1925 +		{
  1.1926 +		process.Kill(0);		// abort 
  1.1927 +		}
  1.1928 +	else
  1.1929 +		{
  1.1930 +		process.Resume();	// logon OK
  1.1931 +		}
  1.1932 +	User::WaitForRequest(status);
  1.1933 +	
  1.1934 +	// we can't use the 'exit reason' if the exe panicked as this
  1.1935 +	// is the panic 'reason' and may be '0' which cannot be distinguished
  1.1936 +	// from KErrNone
  1.1937 +	result = process.ExitType() == EExitPanic ? KErrGeneral : status.Int();
  1.1938 +	process.Close();
  1.1939 +	return result;
  1.1940 +	}
  1.1941 +
  1.1942 +EXPORT_C void BaflUtils::PersistScreenCalibration(const TDigitizerCalibration& aScreenCalibration)
  1.1943 +	{
  1.1944 +	
  1.1945 +	RFs	fs;
  1.1946 +	TInt err = fs.Connect();
  1.1947 +	if (err == KErrNone)
  1.1948 +		{
  1.1949 +		// Setting up drive to store Screen data
  1.1950 +		TDriveUnit systemDrive(static_cast<TInt>(RFs::GetSystemDrive()));
  1.1951 +		TBuf<KMaxDriveName+KScreenCalibrationPathLength>  ScreenFileNameWithDriveAndPath(systemDrive.Name());
  1.1952 +		ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFolder);
  1.1953 +		
  1.1954 +		// Ensure directory \System\Data exists in target drive
  1.1955 +		TRAP(err, EnsurePathExistsL(fs, ScreenFileNameWithDriveAndPath));
  1.1956 +		if(err == KErrNone)
  1.1957 +			{
  1.1958 +			ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFileName);	
  1.1959 +			
  1.1960 +			RFile file;
  1.1961 +			err = file.Replace(fs,ScreenFileNameWithDriveAndPath,EFileWrite|EFileShareExclusive);
  1.1962 +			if (err == KErrNone)
  1.1963 +				{
  1.1964 +				// Write aScreenCalibration to file. 
  1.1965 +				TPtrC8 calptr((const TUint8*)&aScreenCalibration, sizeof(TDigitizerCalibration));
  1.1966 +				err = file.Write(calptr);
  1.1967 +				}
  1.1968 +			file.Close();
  1.1969 +			}
  1.1970 +		}
  1.1971 +	fs.Close();
  1.1972 +	}
  1.1973 +
  1.1974 +EXPORT_C void BaflUtils::InitialiseScreenCalibration(RFs& aFs)
  1.1975 +	{
  1.1976 +	TFindFile ff(aFs);
  1.1977 +	if (ff.FindByDir(KScreenCalibrationFileName, KScreenCalibrationFolder)==KErrNone)
  1.1978 +		{
  1.1979 +		RFile file; 
  1.1980 +		if (file.Open(aFs,ff.File(),EFileRead) == KErrNone )
  1.1981 +			{
  1.1982 +			TDigitizerCalibration calibrationSetting;
  1.1983 +			TPtr8 scrcal((TUint8*)&calibrationSetting, sizeof(TDigitizerCalibration));
  1.1984 +			if( file.Read(scrcal, sizeof( TDigitizerCalibration )) == KErrNone )
  1.1985 +				UserHal::SetXYInputCalibration(calibrationSetting);
  1.1986 +
  1.1987 +			}
  1.1988 +		file.Close();
  1.1989 +		}
  1.1990 +	}
  1.1991 +
  1.1992 +EXPORT_C void BaflUtils::InitialiseHAL(RFs&)
  1.1993 +/** Initialise the HAL settings from. 
  1.1994 +@deprecated 9.1
  1.1995 +This function is empty
  1.1996 +*/
  1.1997 +	{
  1.1998 +	}
  1.1999 +
  1.2000 +EXPORT_C void BaflUtils::InitialiseLocale(RFs& /* aFs */) 
  1.2001 +	{
  1.2002 +// Replaced by new repository based locale initialisation mechanism.
  1.2003 +	}
  1.2004 +
  1.2005 +
  1.2006 +//
  1.2007 +// class CEikFileSortTable
  1.2008 +// 
  1.2009 +
  1.2010 +/**
  1.2011 +*/	
  1.2012 +EXPORT_C CBaflFileSortTable::CBaflFileSortTable()
  1.2013 +	: CArrayFixFlat<TUid>(EArrayGranularity)
  1.2014 +	{}
  1.2015 +
  1.2016 +/**
  1.2017 + * Loads the CBaflFileSortTable using the UIDs read from the TResourceReader supplied in aReader. 
  1.2018 + * @param aReader TResourceReader from which UIDS are read.
  1.2019 + * @leave KErrNoMemory if there is insufficient memory available or one of the system wide error codes.
  1.2020 + */	
  1.2021 +EXPORT_C void CBaflFileSortTable::ConstructFromResourceL(TResourceReader& aReader)
  1.2022 +	{
  1.2023 +	const TInt count=aReader.ReadInt16();
  1.2024 +	for (TInt i=0;i<count;i++)
  1.2025 +		AppendL(TUid::Uid(aReader.ReadInt32()));
  1.2026 +	}
  1.2027 +
  1.2028 +//
  1.2029 +// class BaflDir
  1.2030 +//
  1.2031 +
  1.2032 +#define KSystemDirName _L("System") // Name for System directory
  1.2033 +
  1.2034 +
  1.2035 +void BaflDir::RemoveSystem()
  1.2036 +/*  Remove "System" if in list and it's a directory. */	
  1.2037 +{
  1.2038 +	TInt index;
  1.2039 +	TEntry entry;
  1.2040 +	entry.iName=KSystemDirName;
  1.2041 +	TKeyArrayPak key(_FOFF(TEntry,iName),ECmpFolded);
  1.2042 +	if (iArray->Find(entry,key,index)==KErrNone)
  1.2043 +		{
  1.2044 +		entry=(*iArray)[index];
  1.2045 +		if (entry.IsDir())
  1.2046 +			iArray->Delete(index);
  1.2047 +		}
  1.2048 +	};
  1.2049 +
  1.2050 +/*
  1.2051 +This function gets the element at position "aPos" of aEntries array and inserts 
  1.2052 +it to poition "aNewPos". The element size is "aSize". After the operation the array size
  1.2053 +grows by 1, the element at "aPos" position  moves one position forward.
  1.2054 +This function must be called only from BaflDir::SortByTable() and the insert position
  1.2055 +is always less or equal than the position of the element to be inserted.
  1.2056 +*/
  1.2057 +static void InsertL(CArrayPakFlat<TEntry>* aEntries, TInt aPos, TInt aNewPos, TInt aSize)
  1.2058 +	{
  1.2059 +	__ASSERT_DEBUG(aPos >= aNewPos, User::Invariant());
  1.2060 +	//Expand the array adding one empty entry at "aNewPos" position.
  1.2061 +	aEntries->ExpandL(aNewPos, aSize);
  1.2062 +	//After successfull "ExpandL" call "aPos" must be incremented by 1.
  1.2063 +	//Copy the entry from "aPos + 1" position to "aNewPos" position
  1.2064 +	(*aEntries)[aNewPos] = (*aEntries)[++aPos];
  1.2065 +	}
  1.2066 +
  1.2067 +/**
  1.2068 +Copied from f32file.inl (now moved to f32file_private.h)
  1.2069 +Returns the minimum uncompressed size of the TEntry object, including the valid 
  1.2070 +portion of the name string. The returned value is aligned to 4-byte boundary and
  1.2071 +length includes private members.
  1.2072 +
  1.2073 +@internalTechnology
  1.2074 +@return minimum uncompressed size of TEntry object
  1.2075 +*/
  1.2076 +
  1.2077 +TInt BaflDir::MinEntrySize(const TEntry & aEntry)
  1.2078 +	{
  1.2079 +	return(sizeof(TUint)+sizeof(TInt)+sizeof(TTime)+sizeof(TInt)+sizeof(TUidType)+
  1.2080 +         Align4(aEntry.iName.Size()) + 2*sizeof(TInt));
  1.2081 +	}
  1.2082 +
  1.2083 +TInt BaflDir::SortByTable(CBaflFileSortTable* aTable)
  1.2084 +/**
  1.2085 +Sort into order from given table.
  1.2086 +Any file with uid[2] matching an entry in the table will be sorted relative to 
  1.2087 +others in the table and before any files with no matching uid. 
  1.2088 +For Example: Assume UID table is filled with below 2 UID's
  1.2089 +table[0] =0x10003a64 and table[1] =0x10003a5c. Then file with UID[2]=0x10003a64 
  1.2090 +will be sorted first in the list followed by file with UID[2]=0x10003a5c. Rest
  1.2091 +files will be sorted in the ascending order of UID[2] with directories preceding
  1.2092 +the files.
  1.2093 +
  1.2094 +@param aTable A sort order table containing the UIDs to use in the sort. 
  1.2095 +@return KErrNone if suceessful, otheriwse another system-wide error code.
  1.2096 +*/
  1.2097 +	{
  1.2098 +	TInt r=this->Sort(EDirsFirst|ESortByUid);
  1.2099 +	if(r != KErrNone)
  1.2100 +		{
  1.2101 +		return r;
  1.2102 +		}
  1.2103 +	const TInt tableCount=aTable->Count();
  1.2104 +	const TInt count=iArray->Count();
  1.2105 +	TInt sortedInsertionPoint = 0;
  1.2106 +	for (TInt i=0;i<tableCount;i++)	
  1.2107 +		{
  1.2108 +		TUid	tableUid;
  1.2109 +		// get each UID in the table
  1.2110 +		tableUid = aTable->At(i);
  1.2111 +		for (TInt j=sortedInsertionPoint;j<count;j++)
  1.2112 +			{
  1.2113 +			//parse files in the array list
  1.2114 +			TEntry* pEntry=&(*iArray)[j];
  1.2115 +			// check table UID for match with UID[2] of the file in the list
  1.2116 +			// if found, move the file at the top in the list, followed by next matching UID
  1.2117 +			if (tableUid == pEntry->iType[2])
  1.2118 +				{
  1.2119 +				TRAPD(insertErr, ::InsertL(iArray, j, sortedInsertionPoint++, MinEntrySize(*pEntry)));
  1.2120 +				if(insertErr!=KErrNone)
  1.2121 +					{
  1.2122 +					return insertErr;
  1.2123 +					}
  1.2124 +				iArray->Delete(j+1);
  1.2125 +				}
  1.2126 +			}
  1.2127 +		}
  1.2128 +		iArray->Compress();
  1.2129 +		return KErrNone;
  1.2130 +	}