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 + }