sl@0: // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // Started by DWW, November 1995 sl@0: // BAFL utilities static class sl@0: // sl@0: // sl@0: sl@0: #include "BaUtilsImp.h" sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: /** sl@0: Mimimum length of a filename and mimimum length of a suffix. sl@0: Note these two values are tied together. sl@0: */ sl@0: const TInt KInvNameAndMinSuffixLength = 2; sl@0: sl@0: #define ISDIGIT(c) (c-'0' >= 0 && c-'0' <= 9) sl@0: sl@0: _LIT(KAllDrives, "YXWVUTSRQPONMLKJIHGFEDCBAZ"); sl@0: const TInt KDriveAndPathLength = 3; sl@0: sl@0: // screen calibration stuff sl@0: _LIT(KScreenCalibrationFolder,"\\System\\Data\\"); sl@0: _LIT(KScreenCalibrationFileName, "Screen.DAT"); sl@0: const TInt KScreenCalibrationPathLength = 23; // folder + filename sl@0: sl@0: sl@0: // #define DO_PROFILING sl@0: sl@0: #if defined(DO_PROFILING) sl@0: #pragma message ("------------ N.B. profiling of \"BaflUtils::NearestLanguageFile\" is enabled") sl@0: #include sl@0: #define FIRST_PROFILE_INDEX 50 sl@0: #define PROFILE_INDEX_1 (FIRST_PROFILE_INDEX+0) sl@0: #define PROFILE_COUNT 1 sl@0: #endif sl@0: sl@0: // BaflUtils sl@0: sl@0: class BaflDir : public CDir sl@0: { sl@0: public: sl@0: void RemoveSystem(); sl@0: TInt SortByTable(CBaflFileSortTable* aTable); sl@0: private: sl@0: TInt MinEntrySize(const TEntry & aEntry); sl@0: }; sl@0: sl@0: sl@0: LOCAL_C const TLanguage dp0[] = { ELangCanadianEnglish, ELangAmerican,ELangEnglish, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp1[] = { ELangAmerican, ELangEnglish,ELangCanadianEnglish, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangNone }; sl@0: LOCAL_C const TLanguage dp2[] = { ELangAustralian, ELangEnglish, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp3[] = { ELangSouthAfricanEnglish, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangInternationalEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp4[] = { ELangInternationalEnglish, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp5[] = { ELangEnglish_Apac, ELangEnglish, ELangAustralian, ELangAmerican,ELangInternationalEnglish,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp6[] = { ELangEnglish_Taiwan, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangInternationalEnglish,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp7[] = { ELangEnglish_HongKong, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangInternationalEnglish,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp8[] = { ELangEnglish_Prc, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangInternationalEnglish,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp9[] = { ELangEnglish_Japan, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangInternationalEnglish,ELangEnglish_Thailand,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp10[] = { ELangEnglish_Thailand, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangInternationalEnglish,ELangEnglish_India,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp11[] = { ELangEnglish_India, ELangEnglish, ELangAustralian, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangInternationalEnglish,ELangNewZealand,ELangSouthAfricanEnglish,ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp12[] = { ELangNewZealand, ELangEnglish, ELangAmerican, ELangEnglish_Apac,ELangEnglish_Taiwan,ELangEnglish_HongKong,ELangEnglish_Prc,ELangEnglish_Japan,ELangEnglish_Thailand,ELangEnglish_India,ELangAustralian,ELangInternationalEnglish,ELangSouthAfricanEnglish, ELangCanadianEnglish,ELangNone }; sl@0: LOCAL_C const TLanguage dp13[] = { ELangInternationalFrench,ELangFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone }; sl@0: LOCAL_C const TLanguage dp14[] = { ELangBelgianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangCanadianFrench,ELangNone }; sl@0: LOCAL_C const TLanguage dp15[] = { ELangCanadianFrench, ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangNone }; sl@0: LOCAL_C const TLanguage dp16[] = { ELangFrench,ELangInternationalFrench,ELangSwissFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone }; sl@0: LOCAL_C const TLanguage dp17[] = { ELangSwissFrench,ELangFrench,ELangInternationalFrench,ELangBelgianFrench,ELangCanadianFrench,ELangNone }; sl@0: LOCAL_C const TLanguage dp18[] = { ELangSwissGerman,ELangGerman,ELangAustrian,ELangNone }; sl@0: LOCAL_C const TLanguage dp19[] = { ELangAustrian,ELangGerman,ELangSwissGerman,ELangNone }; sl@0: LOCAL_C const TLanguage dp20[] = { ELangGerman,ELangSwissGerman,ELangAustrian,ELangNone }; sl@0: LOCAL_C const TLanguage dp21[] = { ELangSerbian,ELangCroatian,ELangNone }; sl@0: LOCAL_C const TLanguage dp22[] = { ELangCroatian,ELangSerbian,ELangNone }; sl@0: LOCAL_C const TLanguage dp23[] = { ELangRomanian,ELangMoldavian,ELangNone }; sl@0: LOCAL_C const TLanguage dp24[] = { ELangMoldavian,ELangRomanian,ELangNone }; sl@0: LOCAL_C const TLanguage dp25[] = { ELangBelgianFlemish,ELangDutch,ELangNone }; sl@0: LOCAL_C const TLanguage dp26[] = { ELangDutch,ELangBelgianFlemish,ELangNone }; sl@0: LOCAL_C const TLanguage dp27[] = { ELangAfrikaans,ELangDutch,ELangNone }; sl@0: LOCAL_C const TLanguage dp28[] = { ELangMalay_Apac,ELangMalay,ELangNone }; sl@0: LOCAL_C const TLanguage dp29[] = { ELangIndonesian_Apac,ELangIndonesian,ELangNone }; sl@0: LOCAL_C const TLanguage dp30[] = { ELangSpanish,ELangInternationalSpanish,ELangLatinAmericanSpanish,ELangNone }; sl@0: LOCAL_C const TLanguage dp31[] = { ELangLatinAmericanSpanish,ELangSpanish,ELangInternationalSpanish,ELangNone }; sl@0: LOCAL_C const TLanguage dp32[] = { ELangInternationalSpanish,ELangSpanish,ELangLatinAmericanSpanish,ELangNone }; sl@0: LOCAL_C const TLanguage dp33[] = { ELangCyprusGreek,ELangGreek,ELangNone }; sl@0: LOCAL_C const TLanguage dp34[] = { ELangGreek,ELangCyprusGreek,ELangNone }; sl@0: LOCAL_C const TLanguage dp35[] = { ELangSwissItalian,ELangItalian,ELangNone }; sl@0: LOCAL_C const TLanguage dp36[] = { ELangItalian,ELangSwissItalian,ELangNone }; sl@0: LOCAL_C const TLanguage dp37[] = { ELangBrazilianPortuguese,ELangPortuguese,ELangNone }; sl@0: LOCAL_C const TLanguage dp38[] = { ELangPortuguese,ELangBrazilianPortuguese,ELangNone }; sl@0: LOCAL_C const TLanguage dp39[] = { ELangFinlandSwedish,ELangSwedish,ELangNone }; sl@0: LOCAL_C const TLanguage dp40[] = { ELangSwedish,ELangFinlandSwedish,ELangNone }; sl@0: LOCAL_C const TLanguage dp41[] = { ELangCyprusTurkish,ELangTurkish,ELangNone }; sl@0: LOCAL_C const TLanguage dp42[] = { ELangTurkish,ELangCyprusTurkish,ELangNone }; sl@0: LOCAL_C const TLanguage dp43[] = { ELangHongKongChinese, ELangTaiwanChinese, ELangPrcChinese,ELangNone }; sl@0: LOCAL_C const TLanguage dp44[] = { ELangTaiwanChinese, ELangHongKongChinese,ELangPrcChinese,ELangNone }; sl@0: LOCAL_C const TLanguage dp45[] = { ELangPrcChinese, ELangHongKongChinese, ELangTaiwanChinese,ELangNone }; sl@0: LOCAL_C const TLanguage * const KEquivalentLists[] = { dp0, dp1, dp2, dp3, dp4, dp5, dp6, sl@0: dp7, dp8, dp9, dp10, dp11, dp12, dp13, dp14, dp15, dp16, dp17, sl@0: dp18, dp19, dp20, dp21, dp22, dp23, dp24, dp25, dp26, dp27, sl@0: dp28, dp29, dp30, dp31, dp32, dp33, dp34, dp35, dp36, dp37, sl@0: dp38, dp39, dp40, dp41, dp42, dp43, dp44, dp45}; sl@0: sl@0: /** sl@0: This function gets the list of languages that are 'equivalent' to the sl@0: given language. We say language L1 is equivalent to language L2 if sl@0: speakers of L2 can readily understand L1 without intentional study sl@0: or extraordinary effort. sl@0: sl@0: The equivalence relationship is defined in a static table. Please refer sl@0: to the definition of 'KEquivalentLists' for details. sl@0: Each row in the table is formatted like this: sl@0: @code sl@0: L1, L2, L3, ..., Ln, ELangNone sl@0: @codeend sl@0: In which L2, ..., Ln are equivalents of L1, and ELangNone marks the end of an sl@0: entry. The list is ordered. Compared with L3, L2 is nearer to L1. When choosing sl@0: an equivalent of L1, L2 shall be preferred over L3, L3 shall be preferred sl@0: over L4, and so on. sl@0: L1 is always returned as the ‘nearest equivalent’ of L1 itself. sl@0: sl@0: BaflUtils::NearestLanguageFileV2 searches language specific resource files sl@0: according to the 'equivalent' relationship returned by this function. sl@0: sl@0: @param aLang The language whose equivalents needs to be found out. sl@0: @param aEquivalents On return, this array contains the ordered list of sl@0: languages that are equivalent to the given language. If there is no sl@0: entry for the given language in the table, this array will contain sl@0: two elements on return: the first is the given language itself sl@0: and the 2nd one is ELangNone. For any language that has equivalents sl@0: defined, content of he corresponding entry is returned. sl@0: sl@0: @see BaflUtils::NearestLanguageFileV2 sl@0: */ sl@0: EXPORT_C void sl@0: BaflUtils::GetEquivalentLanguageList(TLanguage aLang, TLanguagePath& aEquivalents) sl@0: { sl@0: aEquivalents[0] = aLang; sl@0: aEquivalents[1] = ELangNone; sl@0: const TInt len = sizeof(KEquivalentLists) / sizeof(KEquivalentLists[0]); sl@0: for (TInt i = 0; i < len; ++i) sl@0: { sl@0: const TLanguage *ptr = KEquivalentLists[i]; sl@0: if (ptr[0] == aLang) sl@0: { sl@0: TInt index = 1; sl@0: while (ELangNone != *ptr) sl@0: { sl@0: aEquivalents[index++] = (TLanguage)*(++ptr); sl@0: } sl@0: aEquivalents[index] = ELangNone; sl@0: break; sl@0: } // end if ptr[0] sl@0: } // end for i sl@0: } sl@0: sl@0: /** sl@0: NearestLanguageFileV2 is very similar to the existing 'NearestLanguageFile' sl@0: function. The only difference between NearestLanguageFile and sl@0: NearestLanguageFileV2 is the order in which language specific sl@0: resource files are searched for. sl@0: NearestLanguageFile searches language specific resource files in the sl@0: order defined by the 'downgrade path' of the given language. Content of the sl@0: downgrade path is dependent on the current active locale, and parts of sl@0: it is runtime configurable. sl@0: NearestLanguageFileV2 searches for language specific resource files sl@0: in the order defined by the 'language equivalence table', which is a sl@0: static data table fixed at build time. There is one entry in the table for sl@0: each language that has one or more equivalents. sl@0: sl@0: @param aFs An active file server session. sl@0: @param aName Name of the language-neutral resource file name which consist of sl@0: an optional drive specification, followed by an optional path name, sl@0: followed by basename for filename, followed by a period and extension. sl@0: On return, in case of a match, this is replaced by the language-specific version sl@0: which consists of the last two characters of the extension plus any preceding sl@0: numeric characters being replaced by the language code. Remains unchanged when there's no match sl@0: @param aLanguage On return, in case of a match, this is replaced by the corresponding language. sl@0: In case of no match, it is set to ELangNone. sl@0: sl@0: @see TLanguage sl@0: @see BaflUtils::GetEquivalentLanguageList sl@0: */ sl@0: EXPORT_C void sl@0: BaflUtils::NearestLanguageFileV2(const RFs& aFs,TFileName& aName, TLanguage& aLanguage) sl@0: { sl@0: TNearestLanguageFileFinder finder(aFs); sl@0: TBool goodSuffix=finder.SetFileName(aName); sl@0: sl@0: // Continue only if the suffix is good. sl@0: if(goodSuffix) sl@0: { sl@0: // add preset customised resource drive to drive list sl@0: // Note that errors returned from AddCustomResourceDrive are ignored. This is because if sl@0: // a custom resource drive has not been found we still want to continue on with searching sl@0: // other drives according to our algorithm sl@0: finder.AddCustomResourceDrive(); sl@0: sl@0: GetEquivalentLanguageList(User::Language(), finder.iPath); sl@0: if (!finder.FindLanguageAndDrive() sl@0: && KErrNone != finder.FindFirstLanguageFileAndDrive()) sl@0: finder.RepairFileName(); sl@0: aLanguage = finder.Language(); sl@0: } sl@0: else sl@0: { sl@0: aLanguage = ELangNone; sl@0: } sl@0: } sl@0: sl@0: // TLibAssocBase sl@0: sl@0: EXPORT_C TLibAssocBase::TLibAssocBase(const RLibrary& aLib,TAny* aPtr) sl@0: : iLibrary(aLib),iPtr(aPtr) sl@0: /** sl@0: Constructs the object taking the specified DLL and a class instance. sl@0: sl@0: @param aLib A reference to a DLL that has already been opened. sl@0: @param aPtr An untyped pointer to an object to be associated with the DLL. sl@0: Typically, this object will have been created using sl@0: the ordinal 1 function from that DLL. */ sl@0: {} sl@0: sl@0: sl@0: sl@0: sl@0: EXPORT_C void TLibAssocBase::Set(const RLibrary& aLib,TAny* aPtr) sl@0: /** sl@0: Implements TLibAssoc::Set(). sl@0: sl@0: @param aLib A reference to a DLL that has already been opened. sl@0: @param aClass A pointer to an object to be associated with the DLL. sl@0: Typically, this object will have been created using sl@0: the ordinal 1 function from that DLL. sl@0: sl@0: @see TLibAssoc::Set */ sl@0: { sl@0: __ASSERT_ALWAYS(iLibrary.Handle()==KNullHandle&&iPtr==NULL,Panic(EBafPanicLibAssocAlreadySet)); sl@0: iLibrary=aLib; sl@0: iPtr=aPtr; sl@0: } sl@0: sl@0: sl@0: sl@0: sl@0: EXPORT_C void TLibAssocBase::DoUnload(TAny* aThis) sl@0: /** sl@0: Calls Close() on the associated DLL. sl@0: sl@0: @param aThis An untyped pointer to a TLibAssoc type. sl@0: */ sl@0: { sl@0: TLibAssocBase& l=*(TLibAssocBase*)aThis; sl@0: l.iPtr=NULL; sl@0: l.iLibrary.Close(); sl@0: } sl@0: sl@0: // sl@0: // class BaflUtils sl@0: // sl@0: EXPORT_C void BaflUtils::CopyWithTruncation(TDes& aDest,const TDesC& aSrc,TChar aTruncationSymbol) sl@0: /** Copies a descriptor, abbreviating it to fit the destination descriptor. sl@0: sl@0: If aSrc is less than the maximum length of aDest, then the string is simply sl@0: copied to aDest. sl@0: sl@0: If this is not so, then the left-most characters of aSrc are copied to aDest, sl@0: up to aDest's maximum length-1. aDest's final character is set to be aTruncationSymbol. sl@0: sl@0: @param aDest On return, the truncated string sl@0: @param aSrc The string to truncate sl@0: @param aTruncationSymbol The truncation character to add */ sl@0: { // static sl@0: TInt maxLength=aDest.MaxLength(); sl@0: if (aSrc.Length()<=maxLength) sl@0: aDest.Copy(aSrc); sl@0: else sl@0: { sl@0: aDest.Copy(aSrc.Left(maxLength-1)); sl@0: aDest.Append(aTruncationSymbol); sl@0: } sl@0: } sl@0: sl@0: EXPORT_C TBool BaflUtils::FileExists(const RFs& aFileSession,const TDesC& aFileName) sl@0: /** Checks if the specified file exists. sl@0: sl@0: @param aFs File server session sl@0: @param aFileName File to check sl@0: @return ETrue if the file exists, otherwise EFalse */ sl@0: { // static sl@0: TEntry entry; sl@0: return(aFileSession.Entry(aFileName,entry)==KErrNone); sl@0: } sl@0: sl@0: EXPORT_C TBool BaflUtils::PathExists(RFs& aFs,const TDesC& aPath) sl@0: /** Tests whether a path exists. sl@0: sl@0: The path should contain a drive letter and a directory, or EFalse is returned. sl@0: EFalse is also returned if it contains a filename or filename extension. sl@0: sl@0: If the path is badly formed, for instance if it contains illegal characters, sl@0: or any directory name consists of a single or double dot, or any directory sl@0: name includes wildcard characters, the function returns EFalse. sl@0: sl@0: @param aFs A connected session with the file server. sl@0: @param aPath The path to test for. It should end in a backslash. sl@0: @return ETrue if the path exists, EFalse if not. EFalse is also returned if the sl@0: specified path is badly formed. */ sl@0: { // static sl@0: TParse parse; sl@0: TInt retcode; sl@0: retcode = parse.Set(aPath, NULL, NULL); sl@0: if (retcode != KErrNone) sl@0: return EFalse; sl@0: if ((! parse.DrivePresent()) || (parse.NameOrExtPresent())) sl@0: return EFalse; sl@0: if (parse.Path().Length() == 0) sl@0: return EFalse; sl@0: TFileName dirName = parse.DriveAndPath(); sl@0: if (dirName.Length() > KMaxFileName) sl@0: return(EFalse); sl@0: RDir dir; sl@0: retcode = dir.Open(aFs,dirName,0); sl@0: if (retcode == KErrNone) sl@0: dir.Close(); sl@0: return (retcode == KErrNone); sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::EnsurePathExistsL(RFs& aFileSession,const TDesC& aFileName) sl@0: /** Makes one or more directories, if they do not already exist. sl@0: sl@0: Any valid path component in the specified path that does not already exist sl@0: is created as a directory. If the specified path already exists, the function sl@0: returns normally. sl@0: sl@0: @param aFs File server session sl@0: @param aFileName Path to ensure exists sl@0: @see RFs::MkDirAll() */ sl@0: { // static sl@0: TInt error=aFileSession.MkDirAll(aFileName); sl@0: if (error!=KErrAlreadyExists) sl@0: User::LeaveIfError(error); sl@0: } sl@0: sl@0: EXPORT_C TPtrC BaflUtils::ExtractAppNameFromFullName(const TFullName &aName) sl@0: /** Gets the application name from a full thread name. sl@0: sl@0: @param aName Thread name sl@0: @return Application name sl@0: @see RThread */ sl@0: { sl@0: // static - return the app name (after first :: before next ::, if any) from a full thread name sl@0: TChar delimiter=':'; sl@0: TInt start=aName.Locate(delimiter); sl@0: if (start<0) sl@0: start=0; // should never happen sl@0: else if (aName.Length()>start+2) sl@0: start+=2; sl@0: TPtrC rest=aName.Mid(start); sl@0: TInt end=rest.Locate(delimiter); sl@0: return end<0 ? rest : rest.Left(end); sl@0: } sl@0: sl@0: LOCAL_C TBool IsLanguageExtended(const TLanguage aLanguage) sl@0: { sl@0: // For compatibility reasons, ELangNone is 0xFFFF. However, it's not an extended language. sl@0: if ((aLanguage==ELangNone) || ((static_cast(aLanguage))<=KDialectMask)) sl@0: return EFalse; sl@0: else sl@0: return ETrue; sl@0: } sl@0: sl@0: sl@0: LOCAL_C TLanguage BaseLanguage(const TLanguage aLanguage) sl@0: { sl@0: if (IsLanguageExtended(aLanguage)) sl@0: return static_cast(aLanguage & KDialectMask); sl@0: else sl@0: return aLanguage; sl@0: } sl@0: sl@0: LOCAL_C TLanguage NextLanguage(TLanguage aLanguage) sl@0: /** Returns the next best language to use after aLanguage, sl@0: based on Symbian's base table of language near-equivalence. sl@0: @internalAll */ sl@0: { sl@0: switch (aLanguage) sl@0: { sl@0: case ELangAustralian: sl@0: case ELangNewZealand: sl@0: case ELangSouthAfricanEnglish: sl@0: case ELangInternationalEnglish: sl@0: case ELangAmerican: sl@0: case ELangEnglish_Apac: sl@0: case ELangEnglish_Taiwan: sl@0: case ELangEnglish_HongKong: sl@0: case ELangEnglish_Prc: sl@0: case ELangEnglish_Japan: sl@0: case ELangEnglish_Thailand: sl@0: return ELangEnglish; sl@0: case ELangCanadianEnglish: sl@0: return ELangAmerican; // 2-stage downgrade sl@0: case ELangSwissFrench: sl@0: case ELangBelgianFrench: sl@0: case ELangInternationalFrench: sl@0: case ELangCanadianFrench: sl@0: return ELangFrench; sl@0: case ELangSwissGerman: sl@0: case ELangAustrian: sl@0: return ELangGerman; sl@0: case ELangInternationalSpanish: sl@0: case ELangLatinAmericanSpanish: sl@0: return ELangSpanish; sl@0: case ELangSwissItalian: sl@0: return ELangItalian; sl@0: case ELangFinlandSwedish: sl@0: return ELangSwedish; sl@0: case ELangCyprusTurkish: sl@0: return ELangTurkish; sl@0: case ELangBelgianFlemish: sl@0: return ELangDutch; sl@0: case ELangHongKongChinese: sl@0: return ELangTaiwanChinese; sl@0: case ELangCyprusGreek: sl@0: return ELangGreek; sl@0: case ELangMalay_Apac: sl@0: return ELangMalay; sl@0: case ELangBrazilianPortuguese: sl@0: return ELangPortuguese; sl@0: default: sl@0: return ELangNone; sl@0: } sl@0: } sl@0: sl@0: sl@0: void AddLanguage(TLanguagePath& aPath, TLanguage aNewLanguage) sl@0: /** Add language to the language path if there is space. sl@0: The first empty slot must have "ELangNone" in it. This will also be true sl@0: on exit. */ sl@0: { sl@0: TLanguage *p = aPath; sl@0: const TLanguage *end = &(aPath[KMaxDowngradeLanguages]); sl@0: while (p != end) sl@0: { sl@0: if (*p == aNewLanguage) sl@0: // language already in list sl@0: break; sl@0: if (*p == ELangNone) sl@0: { sl@0: // found the end of the list sl@0: p[0] = aNewLanguage; sl@0: p[1] = ELangNone; sl@0: break; sl@0: } sl@0: ++p; sl@0: } sl@0: return; sl@0: } sl@0: sl@0: void MakeLanguageDowngradePath(TLanguagePath& aPath, sl@0: TLanguage aCurrent, TLanguage aIdeal, const TLocale& aLocale) sl@0: { sl@0: TInt j = 0; sl@0: if( aIdeal != ELangNone) sl@0: { sl@0: aPath[j++]=aIdeal; sl@0: } sl@0: aPath[j++] = aCurrent; sl@0: aPath[j++] = ELangNone; sl@0: sl@0: if (aCurrent & ~KDialectMask) sl@0: AddLanguage(aPath, static_cast(aCurrent & KDialectMask)); sl@0: sl@0: for (TInt i=0;i<=2;i++) sl@0: { sl@0: AddLanguage(aPath, aLocale.LanguageDowngrade(i)); sl@0: AddLanguage(aPath, BaseLanguage(aLocale.LanguageDowngrade(i))); sl@0: } sl@0: sl@0: while (ELangNone != (aCurrent = NextLanguage(BaseLanguage(aCurrent)))) sl@0: AddLanguage(aPath, aCurrent); sl@0: } sl@0: sl@0: TInt RRealDirectoryScanner::Open(RFs& aFs, const TDesC& aMatchPattern) sl@0: { sl@0: return iDir.Open(aFs, aMatchPattern, sl@0: KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive); sl@0: } sl@0: sl@0: TInt RRealDirectoryScanner::Next(TEntry& aOut) sl@0: { sl@0: return iDir.Read(aOut); sl@0: } sl@0: sl@0: void RRealDirectoryScanner::Close() sl@0: { sl@0: iDir.Close(); sl@0: } sl@0: sl@0: /** sl@0: Simply counts the number of numerical characters at the end of the name passed. sl@0: sl@0: @internalComponent sl@0: @param aFilename The filename to parse sl@0: sl@0: @return Count of the numeric digits at the end of the name passed, sl@0: e.g. x.r491 gives 3. sl@0: */ sl@0: TInt TNearestLanguageFileFinder::CountDigitsFromEnd(const TDesC& aFilename) sl@0: { sl@0: TInt digitCount = 0; sl@0: sl@0: for (TInt idx = aFilename.Length()-1; idx>=0 && ISDIGIT (aFilename [idx]); --idx) sl@0: { sl@0: ++digitCount; sl@0: } sl@0: sl@0: return digitCount; sl@0: } sl@0: sl@0: sl@0: /** sl@0: Counts the number of digits at the end of a filename. sl@0: sl@0: @internalComponent sl@0: @param aFilename The filename to parse sl@0: sl@0: @return Count of the numeric digits at the end of the suffix, sl@0: e.g. x.r491 gives 3. sl@0: 0 if no numeric end of suffix, sl@0: KErrBadName for an invalid filename, sl@0: KErrNotSupported if the filename (minus path) is less sl@0: than or equal to KInvNameAndMinSuffixLength in length sl@0: */ sl@0: TInt TNearestLanguageFileFinder::CountDigitsFromEndInSuffix (const TDesC& aFilename) sl@0: { sl@0: TInt digitCount = 0; sl@0: TInt slashIdx = 0; sl@0: TInt len = aFilename.Length (); sl@0: sl@0: // NOTE: We didn't use TChar here as they are too slow. sl@0: // We also didn't use TParse as they are too large. sl@0: sl@0: // don't work on the path sl@0: for (slashIdx=len-1; slashIdx >= 0 && aFilename[slashIdx] != '\\'; --slashIdx) sl@0: {/*do nothing*/}; sl@0: sl@0: // Get new length sl@0: if (slashIdx>=0) {len = len-slashIdx-1;} sl@0: sl@0: // Initial test to see if filename legal size. sl@0: if (len > KInvNameAndMinSuffixLength) sl@0: { sl@0: digitCount = CountDigitsFromEnd(aFilename); sl@0: sl@0: // Can't store something bigger or we'll panic! sl@0: if (digitCount > KMaxSuffixLength) sl@0: { sl@0: digitCount = KErrBadName; sl@0: } sl@0: else sl@0: // numeric filename, e.g. "1234". sl@0: // No preceeding alpha character sl@0: if (!(len-digitCount)) sl@0: { sl@0: digitCount = KErrBadName; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: digitCount = KErrNotSupported; sl@0: } sl@0: sl@0: return digitCount; sl@0: } sl@0: sl@0: RDirectoryScanner& TNearestLanguageFileFinder::DirectoryScanner() sl@0: { sl@0: return iDirScanner; sl@0: } sl@0: sl@0: TBool TNearestLanguageFileFinder::FileExists(const TDesC& aFileName) const sl@0: { sl@0: return BaflUtils::FileExists(iFs, aFileName); sl@0: } sl@0: sl@0: TBool TNearestLanguageFileFinder::FindDrive() sl@0: { sl@0: ASSERT(iFileName); sl@0: TBool found=EFalse; sl@0: TInt driveLength=iDrives.Length(); sl@0: for (TInt drive = 0; drive!=driveLength; ++drive) sl@0: { sl@0: (*iFileName)[0] = iDrives[drive]; sl@0: if (FileExists(*iFileName)) sl@0: { sl@0: found=ETrue; sl@0: break; sl@0: } sl@0: } sl@0: return found; sl@0: } sl@0: sl@0: TBool TNearestLanguageFileFinder::AppendLanguageCode(TLanguage aLanguage) sl@0: { sl@0: TInt rest = static_cast(aLanguage); sl@0: #ifdef _DEBUG sl@0: _LIT(KErrorMessage, "Bafl"); sl@0: #endif sl@0: __ASSERT_DEBUG(0 <= rest, User::Panic(KErrorMessage,KErrArgument)); sl@0: iFileName->SetLength(iBaseLength); sl@0: const TInt remaining = iFileName->MaxLength() - iBaseLength; sl@0: TInt soFar = 0; sl@0: TBuf<1> num; sl@0: num.Append('0'); sl@0: TBool appendLangSuccess = ETrue; sl@0: TInt digitCount = 0; sl@0: TInt digit = 0; sl@0: while (rest) sl@0: { sl@0: if (remaining == soFar) sl@0: { sl@0: // no more room in descriptor- return rather than panic, sl@0: // file cannot exist. sl@0: iFileName->SetLength(iBaseLength); sl@0: appendLangSuccess= EFalse; sl@0: break; sl@0: } sl@0: // Convert the number to ASCII by consistantly getting the base 10 remainder to convert. sl@0: // The number is updated minus the remainder for the next iteration. sl@0: // eg (rest = 123) -> (12, r3) -> (1, r2) -> (0, r1) sl@0: // Then insert the ASCII representation of the remainder into the filename end sl@0: // so it appears the correct way round. sl@0: // eg (filename.r) -> (filename.r3) -> (filename.r23) -> (filename.r123) sl@0: digit = rest % 10; sl@0: digitCount++; sl@0: rest /= 10; sl@0: num[0] = static_cast(digit + '0'); sl@0: iFileName->Insert(iBaseLength, num); sl@0: sl@0: // Minimum suffix length is KInvNameAndMinSuffixLength sl@0: // so we have to insert zeros to make this up. sl@0: while (!rest && digitCount < KInvNameAndMinSuffixLength) sl@0: { sl@0: num[0] = static_cast('0'); sl@0: iFileName->Insert(iBaseLength, num); sl@0: ++digitCount; sl@0: } sl@0: sl@0: ++soFar; sl@0: } sl@0: sl@0: return appendLangSuccess; sl@0: } sl@0: sl@0: sl@0: TBool TNearestLanguageFileFinder::FindLanguageAndDrive() sl@0: /** Search for files across all drives in all languages in the path plus the sl@0: language-neutral file. */ sl@0: { sl@0: ASSERT(iFileName); sl@0: // No point appending if the suffix is bad sl@0: for (const TLanguage* currentLang = iPath; *currentLang != ELangNone; ++currentLang) sl@0: { sl@0: if (AppendLanguageCode(*currentLang) && FindDrive()) sl@0: { sl@0: iLanguage = *currentLang; sl@0: return ETrue; sl@0: } sl@0: } sl@0: // search for language-neutral file sl@0: iFileName->SetLength(iBaseLength); sl@0: iFileName->Append(iSuffix); sl@0: return FindDrive(); sl@0: } sl@0: sl@0: TInt TNearestLanguageFileFinder::LanguageNumberFromFile(const TDesC& aFileName, const TDesC& aStem) sl@0: { sl@0: TInt lang = 0; sl@0: TInt multiplier = 1; sl@0: TInt leadingZeroCount = 0; sl@0: TInt languageNumber = KErrNotFound; sl@0: const TText* firstChar = aFileName.Ptr(); sl@0: const TText* lastChar = firstChar + aFileName.Length() - 1; sl@0: const TText* currentChar = lastChar; sl@0: // string cannot contain only numbers, because it must have a ':' in it sl@0: while ('0' <= *currentChar && *currentChar <= '9') sl@0: { sl@0: if (*currentChar == '0') sl@0: leadingZeroCount++; sl@0: else sl@0: { sl@0: leadingZeroCount = 0; sl@0: lang += multiplier * (*currentChar - '0'); sl@0: } sl@0: multiplier *= 10; sl@0: --currentChar; sl@0: } sl@0: TInt along=lastChar - currentChar; sl@0: if (2 <= along) sl@0: { sl@0: // We have at least 2 digits at the end. sl@0: // trim of bad leading zeros sl@0: TInt maxTrim = along - 2; sl@0: if (maxTrim < leadingZeroCount) sl@0: { sl@0: leadingZeroCount = maxTrim; sl@0: } sl@0: currentChar += leadingZeroCount; sl@0: // we have at least 2 digits at the end but does the rest of it match the stem? sl@0: TPtrC foundStem(firstChar, currentChar - firstChar + 1); sl@0: //foundStem.CompareF(aStem.Right(foundStem.Length())) sl@0: if (0 == foundStem.CompareF(aStem)) sl@0: { sl@0: languageNumber=lang; sl@0: } sl@0: } sl@0: return languageNumber; sl@0: } sl@0: sl@0: TInt TNearestLanguageFileFinder::FindFirstLanguageFile(RFs& aFs) sl@0: { sl@0: ASSERT(iFileName); sl@0: iFileName->SetLength(iBaseLength); sl@0: TPtrC name(*iFileName); sl@0: TParsePtrC nameToParse(name); sl@0: TPtrC nameStem(nameToParse.NameAndExt()); sl@0: iFileName->Append('*'); sl@0: TInt bestLanguageMatch = KMaxTInt; sl@0: RDirectoryScanner& scanner = DirectoryScanner(); sl@0: TInt err = scanner.Open(aFs, *iFileName); sl@0: if (err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: TEntry entry; sl@0: while (KErrNone == scanner.Next(entry)) sl@0: { sl@0: TInt lang = LanguageNumberFromFile(entry.iName, nameStem); sl@0: if (0 < lang && lang < bestLanguageMatch) sl@0: { sl@0: bestLanguageMatch = lang; sl@0: } sl@0: } sl@0: scanner.Close(); sl@0: if (bestLanguageMatch != KMaxTInt) sl@0: { sl@0: iLanguage = static_cast(bestLanguageMatch); sl@0: AppendLanguageCode(static_cast(bestLanguageMatch)); sl@0: return KErrNone; sl@0: } sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: // Try each drive for any language files sl@0: // iFileName must have a directory specifier sl@0: TInt TNearestLanguageFileFinder::FindFirstLanguageFileAndDrive() sl@0: { sl@0: ASSERT(iFileName); sl@0: TInt findFirstResult=KErrNotFound; sl@0: TInt driveLength=iDrives.Length(); sl@0: for (TInt drive = 0; drive != driveLength; ++drive) sl@0: { sl@0: (*iFileName)[0] = iDrives[drive]; sl@0: TInt err = FindFirstLanguageFile(CONST_CAST(RFs&,iFs)); sl@0: if (err == KErrNone || err == KErrNoMemory) sl@0: { sl@0: findFirstResult=err; sl@0: break; sl@0: } sl@0: } sl@0: return findFirstResult; sl@0: } sl@0: sl@0: /** sl@0: Invalid filenames are any filename whose length (minus path) must be greater sl@0: than KInvNameAndMinSuffixLength, and whose form is purely numerical, i.e. '1234' sl@0: */ sl@0: TBool TNearestLanguageFileFinder::SetFileName(TFileName& aFileName) sl@0: { sl@0: iDrives.Zero(); sl@0: iFileName = &aFileName; sl@0: iOriginalBaseLength = iFileName->Length(); sl@0: sl@0: TInt suffixLength = CountDigitsFromEndInSuffix (aFileName); sl@0: sl@0: // No point trying for filenames thats are badly formed sl@0: // or that are too large. sl@0: if (suffixLength >= 0 && sl@0: KInvNameAndMinSuffixLength < iOriginalBaseLength) sl@0: { sl@0: if (suffixLength > 0) sl@0: { sl@0: // all of suffix to be replaced sl@0: iSuffix = iFileName->Right(suffixLength); sl@0: iOriginalBaseLength -= suffixLength; sl@0: iFileName->SetLength(iOriginalBaseLength); sl@0: } sl@0: else sl@0: { sl@0: // No numerical part to suffix sl@0: TInt periodIdx = 0; sl@0: sl@0: // Search for the period within range KInvNameAndMinSuffixLength sl@0: // from the end. As this must work for all values of sl@0: // KInvNameAndMinSuffixLength sl@0: for (TInt i = iOriginalBaseLength-1; sl@0: !periodIdx && i >= (iOriginalBaseLength-KInvNameAndMinSuffixLength-1); sl@0: --i) sl@0: { sl@0: if ((*iFileName) [i] == '.') sl@0: { sl@0: periodIdx = i; sl@0: } sl@0: } sl@0: sl@0: // Don't handle files ending in a period. sl@0: // This is because the behaviour is different between Windows sl@0: // and Symbian Fs. In Windows it strips the period off. sl@0: // sl@0: // However, and this shouldn't happen as it is not shown sl@0: // (in the documentation) to be valid. sl@0: // Just try our best. sl@0: if (periodIdx == iOriginalBaseLength-1) sl@0: { sl@0: iSuffix.Zero(); sl@0: return EFalse; sl@0: } sl@0: else sl@0: if (periodIdx) sl@0: { sl@0: // If there are KInvNameAndMinSuffixLength chars after the period sl@0: // simply replace them. sl@0: TInt right = iOriginalBaseLength-periodIdx-1; sl@0: iSuffix = iFileName->Right(right); sl@0: iOriginalBaseLength -= right; sl@0: iFileName->SetLength(iOriginalBaseLength); sl@0: } sl@0: else sl@0: { sl@0: // Make the suffix start from KInvNameAndMinSuffixLength sl@0: // from the right sl@0: TInt right = KInvNameAndMinSuffixLength; sl@0: iSuffix = iFileName->Right(right); sl@0: iOriginalBaseLength -= right; sl@0: iFileName->SetLength(iOriginalBaseLength); sl@0: } sl@0: } sl@0: } sl@0: else sl@0: { sl@0: // bad or no suffix - treat the same sl@0: iSuffix.Zero(); sl@0: return EFalse; sl@0: } sl@0: sl@0: // For filenames with no drive letter prefix and also for filenames sl@0: // shorter than the drive letter length, i.e. with no drive sl@0: // information, insert it. sl@0: // Handles if the user simply enters the drive, e.g. "c:". sl@0: if (iOriginalBaseLength < KMaxDriveName || (*iFileName)[1] != ':') sl@0: { sl@0: // Set up the default if none supplied and make room in the filename sl@0: // array to contain a drive specification. Set initial drive letter to -1 sl@0: // so the iFileName is repaired before exited sl@0: iInitialDriveLetter = -1; sl@0: iFileName->Insert(0, _L("_:")); sl@0: iDrives.Append('Z'); sl@0: } sl@0: else sl@0: { sl@0: // Use the drive supplied inthe aName to NearestLanguageFile() sl@0: iInitialDriveLetter = (*iFileName)[0]; sl@0: iDrives.Append(iInitialDriveLetter); sl@0: } sl@0: sl@0: iBaseLength = iFileName->Length(); sl@0: sl@0: return ETrue; sl@0: } sl@0: sl@0: sl@0: TLanguage TNearestLanguageFileFinder::Language() sl@0: { sl@0: return iLanguage; sl@0: } sl@0: sl@0: TNearestLanguageFileFinder::TNearestLanguageFileFinder( sl@0: const RFs& aFs) sl@0: : iFs(aFs), iFileName(0), iLanguage(ELangNone) sl@0: { sl@0: } sl@0: sl@0: void TNearestLanguageFileFinder::RepairFileName() sl@0: { sl@0: ASSERT(iFileName); sl@0: iFileName->SetLength(iBaseLength); sl@0: if (iInitialDriveLetter == -1) sl@0: iFileName->Delete(0, 2); sl@0: else sl@0: (*iFileName)[0] = static_cast(iInitialDriveLetter); sl@0: iFileName->SetLength(iOriginalBaseLength); sl@0: iFileName->Append(iSuffix); sl@0: } sl@0: sl@0: sl@0: /** sl@0: Add the custom resource drive to the start of the iDrives string. sl@0: sl@0: The custom resource drive is a preset writeable drive on which customised sl@0: resource files may be present. This drive takes priority over the other sl@0: drives when searching for language files. sl@0: sl@0: @return KErrNone if iDrives string was successfully modified; KErrAlreadyExists sl@0: if the drive is already present in the string; otherwise one of sl@0: the other system-wide error codes (iDrives will be unmodified). sl@0: */ sl@0: TInt TNearestLanguageFileFinder::AddCustomResourceDrive() sl@0: { sl@0: TInt drive = GetCustomResourceDriveNumber(); sl@0: if (drive<0) sl@0: return drive; sl@0: sl@0: // if drive not already in drive list sl@0: if (iDrives.LocateF('A' + drive) < 0) sl@0: { sl@0: // add it sl@0: _LIT(KDrivePlaceholder, "_"); sl@0: iDrives.Insert(0, KDrivePlaceholder); sl@0: iDrives[0] = 'A' + drive; sl@0: return KErrNone; sl@0: } sl@0: else sl@0: return KErrAlreadyExists; sl@0: } sl@0: sl@0: sl@0: void TNearestLanguageFileFinder::AddAllDrives() sl@0: { sl@0: ASSERT(iDrives.Length() < 2); sl@0: if (iDrives.Length() == 0) sl@0: { sl@0: iDrives = KAllDrives; sl@0: return; sl@0: } sl@0: TInt pos = KAllDrives().LocateF(iDrives[0]); sl@0: if (pos < 0) sl@0: { sl@0: iDrives = KAllDrives; sl@0: return; sl@0: } sl@0: iDrives.Append(KAllDrives().Left(pos)); sl@0: iDrives.Append(KAllDrives().Mid(pos + 1)); sl@0: } sl@0: sl@0: sl@0: /** sl@0: Get the value of the custom resource drive. sl@0: sl@0: The custom resource drive is a preset writeable drive on which customised language resource sl@0: files can reside. The drive number is accessed via the HAL attribute ECustomResourceDrive. sl@0: It is then returned if it has been defined as a valid drive no. sl@0: Otherwise for backward compatibility reasons an attempt is then made to access the system sl@0: drive HAL attribute instead. This drive number is returned if it has been defined as a valid sl@0: drive number. sl@0: Otherwise if neither a valid ECustomResourceDrive or ESystemDrive exists then KErrNotFound sl@0: is returned. sl@0: sl@0: Note that the ESystemDrive HAL attribute has been deprecated. It is accessed here to cater sl@0: for existing implementations which still expect it to be used. sl@0: sl@0: @return The drive number (corresponding to a TDriveNumber value) if successful; sl@0: KErrNotFound if neither a valid ECustomResourceDrive or a valid ESystemDrive HAL attribute sl@0: is defined; sl@0: sl@0: @see HAL::ECustomResourceDrive sl@0: @see HAL::ESystemDrive sl@0: */ sl@0: TInt TNearestLanguageFileFinder::GetCustomResourceDriveNumber() const sl@0: { sl@0: TInt drive = KErrNotFound; sl@0: sl@0: // access custom resource drive attribute sl@0: if (HAL::Get(HAL::ECustomResourceDrive, drive) == KErrNone) sl@0: { sl@0: // check that drive is valid sl@0: if (drive>=EDriveA && drive<=EDriveZ) sl@0: return drive; sl@0: } sl@0: sl@0: // access system drive attribute sl@0: // (Note that ESystemDrive is deprecated. It is checked here sl@0: // solely for backward compatibility reasons.) sl@0: if (HAL::Get(HAL::ESystemDrive, drive) == KErrNone) sl@0: { sl@0: // check that drive is valid sl@0: if (drive>=EDriveA && drive<=EDriveZ) sl@0: return drive; sl@0: } sl@0: sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: sl@0: sl@0: /** Get the value of the system drive. sl@0: sl@0: The system drive can be set to one of the built-in read/write drives. Which sl@0: drive is used is hardware-dependent. On some hardware, there may not be a sl@0: system drive. The system drive is used as the drive on which localisable files sl@0: are searched for. This enables a phone to be localised dynamically, using sl@0: files not in the ROM. sl@0: sl@0: @param aDriveNumber On return, contains the drive number of the system drive. sl@0: @return KErrNone is always returned. sl@0: sl@0: @deprecated This method has been replaced by (and now internally calls) sl@0: RFs:GetSystemDrive, which always returns a valid drive number. sl@0: sl@0: @see BaflUtils::NearestLanguageFile sl@0: @see RFs::GetSystemDrive sl@0: */ sl@0: EXPORT_C TInt BaflUtils::GetSystemDrive(TDriveNumber& aDriveNumber) sl@0: { sl@0: aDriveNumber = RFs::GetSystemDrive(); sl@0: return KErrNone; sl@0: } sl@0: sl@0: sl@0: /** Set most appropriate extension language code for filename and set corresponding language. sl@0: sl@0: Symbian uses numeric values to identify natural languages as specified by the TLanguage enumeration sl@0: defined in e32const.h. These values are used at the end of filename extensions to identify the sl@0: languages pertaining to files which have language specific variants such as resource files. sl@0: For instance filename.r01 and filename.r02 would be the English and French versions of the sl@0: resource file filename.rsc. Language codes can be between 2 to 5 digits in length. sl@0: sl@0: Starting from Symbian OS v7.0 this function constructs and uses a language downgrade path which sl@0: consists of up to sixteen TLanguage values the first of which is the ideal language followed by sl@0: the language of the current locale. Up to the next three can be customised using sl@0: TLocale::SetLanguageDowngrade(). The rest of the language downgrade path is based on a sl@0: table of language near equivalence which is internal to Symbian. sl@0: sl@0: This function searches the custom resource drive (if set, retrieved from HAL) sl@0: and then searches the optional drive specified in aName or 'Z:' if none is sl@0: specified in aName. The custom resource drive is retrieved from the HAL sl@0: attribute ECustomResourceDrive if set, if not set it will retrieve the legacy sl@0: value set in the legacy HAL attribute ESystemDrive. No custom resource drive sl@0: is searched if neither are set. sl@0: Note - setting the custom resource drive will reduce the performance of this sl@0: routine which will adversely affect device performance e.g. at boot up. sl@0: On NAND Flash based devices use of a composite Z: drive file system made up of multiple sl@0: ROM images is the preferred mechanism for customising language resources on devices in sl@0: Symbian OS 9.2 onwards, see Developer Library » Base Porting Guide » Porting: background sl@0: information » NAND flash » NAND Flash image format. Thus use of the custom resource drive sl@0: HAL attribute is effectively obsolete. sl@0: sl@0: The last two characters of aName are removed along with any digits which appear before them. sl@0: Then language codes specified in the constructed language downgrade path are appended in turn to sl@0: aName as a match is searched for in the file system. In case no match is found using the sl@0: constructed language downgradepath then files in the specified directory are searched for a sl@0: suitable extension with preference given to the one specified if present. In cases where a sl@0: match takes place the aName and aLanguage arguments are updated otherwise aName is left sl@0: unchanged and aLanguage is set to ELangNone. sl@0: sl@0: Here are some examples of correct and incorrect function usage with different aName inputs, sl@0: file system state and downgrade paths as follows: sl@0: sl@0: @code sl@0: Following files exist: sl@0: C:\\abc.rsc - Language Neutral resource file. sl@0: C:\\abc.r01 - Resource file for the English language. sl@0: C:\\abc.r10 - Resource file for the American-English language. sl@0: C:\\abc.r160 - Resource file for the English as appropriate in Japan. sl@0: sl@0: Constructed Downgrade Language Path cases: sl@0: - Case 1. (ELangAmerican -> ELangEnglish -> ELangNone). sl@0: - Case 2. (ELangEnglish_Japan -> ELangEnglish -> ELangNone). sl@0: - Case 3. Same as case 1, However "C:\\abc.r10" is deleted prior to the function call. sl@0: - Case 4. Same as case 1, However both "C:\\abc.r01" and "C:\\abc.r10" are deleted prior to the function call. sl@0: @endcode sl@0: sl@0: @code sl@0: Input aName . . . . Output aName. . . aLanguage . . . . . Description sl@0: -------------------------------------------------------------------------------------------------------------------- sl@0: "C:\\abc.rsc" . . . "C:\\abc.r10" . . ELangAmerican . . . Match on first language (Case 1) sl@0: "C:\\abc.r10" . . . "C:\\abc.r10" . . ELangAmerican . . . Match, However it's not the intended use of sl@0: . . . . . . . . . . . . . . . . . . . . . . . . . . . . . the function (Case 1) sl@0: "C:\\abc.r" . . . . "C:\\abc.r" . . . ELangNone . . . . . The no. of characters in the suffix is less than sl@0: . . . . . . . . . . . . . . . . . . . . . . . . . . . . . KInvNameAndMinSuffixLength(2)(Case 1) sl@0: "C:\\abc.". . . . . "C:\\abc.". . . . ELangNone . . . . . Invalid Suffix: The filename ends with a period(Case 1) sl@0: "C:\\abc.r123456" . "C:\\abc.r123456" ELangNone . . . . . Invalid Suffix: The no. of digits in the suffix is greater sl@0: . . . . . . . . . . . . . . . . . . . . . . . . . . . . . than KMaxSuffixLength(5) (Case 1) sl@0: "C:\\abc.10". . . . "C:\\abc.10 . . . ELangNone . . . . . Invalid Suffix: There's no proceeding alphabetical sl@0: . . . . . . . . . . . . . . . . . . . . . . . . . . . . . characters in the suffix (Case 1) sl@0: "\\abc.rsc" . . . . "\\abc.rsc" . . . ELangNone . . . . . No drive so Z: search, no match (Case 1) sl@0: "C:\\abc.rsc" . . . "C:\\abc.r160". . ELangEnglish_Japan. Match for language file 3 digits long (Case 2) sl@0: "C:\\abc.rsc" . . . "C:\\abc.r01" . . ELangEnglish. . . . Match on second language (Case 3) sl@0: "C:\\abc.rsc" . . . "C:\\abc.rsc" . . ELangNone . . . . . No corresponding langauge file match found (Case 4) sl@0: --------------------------------------------------------------------------------------------------------------------- sl@0: @endcode sl@0: sl@0: @param aFs File server session. sl@0: @param aName Optional drive specification, followed by optional path name, sl@0: followed by basename for filename, followed by period and extension. sl@0: On return, in case of a match, this is replaced by the language-specific version sl@0: which consists of the last two characters of the extension plus any preceding sl@0: numeric characters being replaced by the language code. Remains unchanged when there's no match sl@0: @param aLanguage On return, in case of a match, this is replaced by the corresponding language. sl@0: In case of no match it's set to ELangNone. sl@0: @see TLanguage sl@0: @see BaflUtils::GetDowngradePathL sl@0: @see TLocale::SetLanguageDowngrade sl@0: */ sl@0: EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage) sl@0: { sl@0: #if defined(DO_PROFILING) sl@0: RDebug::ProfileReset(FIRST_PROFILE_INDEX, PROFILE_COUNT); sl@0: RDebug::ProfileStart(PROFILE_INDEX_1); sl@0: #endif sl@0: TNearestLanguageFileFinder finder(aFs); sl@0: TBool goodSuffix=finder.SetFileName(aName); sl@0: sl@0: // Only continue if the suffix is good. sl@0: if(goodSuffix) sl@0: { sl@0: // add preset customised resource drive to drive list sl@0: // Note that errors returned from AddCustomResourceDrive are ignored. This is because if sl@0: // a custom resource drive has not been found we still want to continue on with searching sl@0: // other drives according to our algorithm sl@0: finder.AddCustomResourceDrive(); sl@0: sl@0: TLocale locale; sl@0: TLanguage idealLanguage; sl@0: idealLanguage = IdealLanguage(); sl@0: MakeLanguageDowngradePath(finder.iPath, User::Language(), idealLanguage, locale); sl@0: if (!finder.FindLanguageAndDrive() sl@0: && KErrNone != finder.FindFirstLanguageFileAndDrive()) sl@0: finder.RepairFileName(); sl@0: aLanguage = finder.Language(); sl@0: } sl@0: sl@0: #if defined(DO_PROFILING) sl@0: RDebug::ProfileEnd(PROFILE_INDEX_1); sl@0: TProfile profile[PROFILE_COUNT]; sl@0: RDebug::ProfileResult(&profile[0], FIRST_PROFILE_INDEX, PROFILE_COUNT); sl@0: if(goodSuffix) sl@0: { sl@0: RDebug::Print(_L("BaflUtils::NearestLanguageFile profile info: %d.%03ds"), profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime/1000000, profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime%1000000); sl@0: } sl@0: else sl@0: { sl@0: RDebug::Print(_L("BaflUtils::NearestLanguageFile (bad suffix ) profile info: %d.%03ds"), profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime/1000000, profile[PROFILE_INDEX_1-FIRST_PROFILE_INDEX].iTime%1000000); sl@0: } sl@0: #endif sl@0: } sl@0: sl@0: /** Searches for the file with the correct language extension for the language sl@0: of the current locale, or failing this, the best matching file. sl@0: sl@0: @param aFs File server session. sl@0: @param aName File name as it would be without a language-specific extension. sl@0: On return, this is changed to the language-specific version. If no such file sl@0: is found, the name is unchanged. sl@0: @see BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName, TLanguage& aLanguage) */ sl@0: EXPORT_C void BaflUtils::NearestLanguageFile(const RFs& aFs,TFileName& aName) sl@0: { sl@0: TLanguage language; sl@0: sl@0: NearestLanguageFile( aFs, aName, language); sl@0: sl@0: (void)language; sl@0: } sl@0: sl@0: /** Set the ideal language for the thread. sl@0: This interface is intended for the use of UIKON only. sl@0: sl@0: @param aLanguage Ideal language. sl@0: @return KErrNone, if successful; KErrNoMemory if there is not enough memory @see TLanguage sl@0: @see BaflUtils::NearestLanguageFile() sl@0: @internalAll */ sl@0: EXPORT_C TInt BaflUtils::SetIdealLanguage(TLanguage aLanguage) sl@0: { sl@0: TLanguage* langPtr=(TLanguage*)Dll::Tls(); sl@0: if( langPtr==NULL) sl@0: { sl@0: langPtr=(TLanguage*)User::Alloc(sizeof(TLanguage)); sl@0: sl@0: if(!langPtr) sl@0: return(KErrNoMemory); sl@0: sl@0: TInt ret=Dll::SetTls(langPtr); sl@0: sl@0: if(ret!=KErrNone) sl@0: return(ret); sl@0: } sl@0: *langPtr=aLanguage; sl@0: return(KErrNone); sl@0: } sl@0: sl@0: /** Get the ideal language of the thread. sl@0: This interface is intended for the use of UIKON only. sl@0: sl@0: @return Ideal language if set, ELangNone if not set sl@0: @see BaflUtils::NearestLanguageFile() sl@0: @internalAll */ sl@0: EXPORT_C TLanguage BaflUtils::IdealLanguage() sl@0: { sl@0: TLanguage* langPtr=(TLanguage*)Dll::Tls(); sl@0: sl@0: if( langPtr==NULL) sl@0: { sl@0: return(ELangNone); sl@0: } sl@0: sl@0: return(*langPtr); sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::ReleaseIdealLanguage() sl@0: /** Releases the ideal language store if it has been allocated. sl@0: This interface is intended for the use of UIKON only. sl@0: sl@0: @internalAll */ sl@0: { sl@0: TLanguage* aLanguage=(TLanguage*)Dll::Tls(); sl@0: if( aLanguage==NULL) sl@0: return; sl@0: sl@0: delete aLanguage; sl@0: Dll::FreeTls(); sl@0: } sl@0: sl@0: EXPORT_C TInt BaflUtils::IsFolder(const RFs& aFs, const TDesC& aFullName, TBool& aIsFolder) sl@0: /** Checks if the specified item is a folder. sl@0: sl@0: @param aFs File server session sl@0: @param aFullName Name to check sl@0: @param aIsFolder ETrue if aFullName is a folder, otherwise EFalse sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes */ sl@0: { sl@0: TParsePtrC parse(aFullName); sl@0: if ((parse.DriveAndPath().Length() == KDriveAndPathLength) && (aFullName.Length() == KDriveAndPathLength)) sl@0: { sl@0: aIsFolder = ETrue; sl@0: return(KErrNone); sl@0: } sl@0: TEntry entry; sl@0: TInt retcode = aFs.Entry(aFullName, entry); sl@0: if (retcode == KErrNone) sl@0: aIsFolder = ((entry.iAtt & KEntryAttDir)==KEntryAttDir); sl@0: sl@0: return(retcode); sl@0: } sl@0: sl@0: sl@0: EXPORT_C TBool BaflUtils::FolderExists(RFs& aFs, const TDesC& aFolderName) sl@0: /** Tests whether a folder exists. sl@0: sl@0: The folder is specified in a path. The path can omit the drive letter, in sl@0: which case the drive letter is taken from the session path. sl@0: sl@0: If the path is badly formed, for instance if it contains illegal characters, sl@0: or any directory name consists of a single or double dot, or any directory sl@0: name includes wildcard characters, the function returns EFalse. sl@0: sl@0: If a filename is included in the path, it is ignored (the existence sl@0: of the file will not be checked). However if included, it must not sl@0: be badly formed - this will cause the function to return EFalse. sl@0: If no filename is specified, the path should end in a backslash. sl@0: sl@0: Examples of valid paths (returning ETrue): sl@0: C:\; \; C:\Readme.txt; C:\system\data\; \system\data\Anyfile.dat sl@0: sl@0: Examples of invalid paths (returning EFalse): sl@0: C:\FolderDoesntExist\; ..\system\; C:\Wild*card\; C:\system\data\Bad>File.txt sl@0: sl@0: @param aFs A connected session with the file server. sl@0: @param aFolderName A path specifying the folder to test for. sl@0: @return ETrue if the folder specified in aFolderName exists, EFalse if not. sl@0: EFalse is also returned if the specified path is badly formed. */ sl@0: { sl@0: if (aFolderName.Length()==0) sl@0: {return EFalse;} sl@0: sl@0: TParse parse; sl@0: sl@0: TInt retcode = parse.SetNoWild(aFolderName, NULL, NULL); sl@0: sl@0: if (retcode != KErrNone) sl@0: {return EFalse;} sl@0: sl@0: if (parse.NameOrExtPresent()) sl@0: if (!aFs.IsValidName(aFolderName)) sl@0: {return EFalse;} sl@0: sl@0: TPtrC dirName = parse.DriveAndPath(); sl@0: RDir dir; sl@0: retcode = dir.Open(aFs,dirName,0); sl@0: if (retcode == KErrNone) sl@0: {dir.Close();} sl@0: return (retcode == KErrNone); sl@0: } sl@0: sl@0: sl@0: EXPORT_C TFileName BaflUtils::FolderNameFromFullName(const TDesC& aFullName) sl@0: /** Gets the folder name from a path. sl@0: sl@0: A valid full name should have a drive associated with it sl@0: e.g ("a:\\" - "z:\\")("a:" - "z:")("c:\\system\data\file.txt") sl@0: Invalid entry will have no drive and cause a panic EBafPanicBadOpenArg sl@0: For example, if the path is "c:\documents\word\mydoc1", then "word" is returned. sl@0: "c:" then "c:" is returned sl@0: "c:\\" then "c:\" is returned sl@0: "c:\\mydoc1.txt then "c:\" is returned sl@0: sl@0: @param aFullName A path. sl@0: @return The folder name. */ sl@0: { sl@0: TParsePtrC parse(aFullName); sl@0: sl@0: __ASSERT_ALWAYS(parse.DrivePresent(),Panic(EBafPanicBadOpenArg)); sl@0: sl@0: TFileName folderName = parse.Path(); sl@0: //If the path name has no associated path(e.g "c:") or path='\'(e.g "c:\\", "c:\\file.txt") sl@0: //then the folder name is just equal to drivename sl@0: sl@0: TBuf<1> pathSeparator; sl@0: pathSeparator.Append(KPathDelimiter); sl@0: sl@0: if (folderName.Length()==0 || folderName==pathSeparator) sl@0: return (parse.DriveAndPath()); sl@0: //else just get the foldername sl@0: TInt len = folderName.Length(); sl@0: TInt pos = --len; sl@0: while (--pos) sl@0: if (folderName.Mid(pos, 1)==pathSeparator) sl@0: break; sl@0: folderName.Delete(len, 1); sl@0: folderName.Delete(0, pos+1); sl@0: return(folderName); sl@0: } sl@0: sl@0: sl@0: EXPORT_C TFileName BaflUtils::DriveAndPathFromFullName(const TDesC& aFullName) sl@0: /** Gets the drive letter and path from a file name. sl@0: sl@0: This is in the form: drive-letter:\\path\\. The drive letter is folded using sl@0: class TCharF. sl@0: sl@0: @param aFullName File name sl@0: @return The drive and path */ sl@0: { sl@0: TParsePtrC parse(aFullName); sl@0: return (parse.DriveAndPath()); sl@0: } sl@0: sl@0: sl@0: EXPORT_C TFileName BaflUtils::RootFolderPath(const TBuf<1> aDriveLetter) sl@0: /** Gets the root folder for the specified drive. sl@0: sl@0: If aDriveLetter is an alphabet(lowercase or uppercase) then it will return sl@0: the TFileName which is simply the drive letter plus ":\" sl@0: If this is not the case, the function will panic with panic code EBafPanicBadOpenArg sl@0: sl@0: @param aDriveLetter Drive letter sl@0: @return Root folder */ sl@0: { sl@0: TChar driveLetter(aDriveLetter[0]); sl@0: driveLetter.LowerCase(); sl@0: TInt aDriveNumber=driveLetter-TChar('a'); sl@0: __ASSERT_ALWAYS(aDriveNumber>= EDriveA && aDriveNumber <= EDriveZ,Panic(EBafPanicBadOpenArg)); sl@0: sl@0: TFileName rootFolderPath = aDriveLetter; sl@0: rootFolderPath.Append(_L(":\\")); sl@0: return rootFolderPath; sl@0: } sl@0: sl@0: sl@0: EXPORT_C void BaflUtils::AbbreviateFileName(const TFileName& aOriginalFileName, TDes& aAbbreviatedFileName) sl@0: /** Abbreviates a file name. sl@0: sl@0: If aOriginalFileName is less than the maximum length of aAbbreviatedFileName, sl@0: then the name is simply copied to aAbbreviatedFileName. sl@0: sl@0: If this is not so, then the left-most characters of aOriginalFileName are sl@0: copied to aAbbreviatedFileName, up to aAbbreviatedFileName's maximum length-1. sl@0: aAbbreviatedFileName's first character is set to be an ellipsis. sl@0: sl@0: @param aOriginalFileName Original file name sl@0: @param aAbbreviatedFileName On return, abbreviated file name */ sl@0: { sl@0: TInt maxWidthInChars = aAbbreviatedFileName.MaxLength(); sl@0: if (aOriginalFileName.Length() <= maxWidthInChars) sl@0: { sl@0: aAbbreviatedFileName = aOriginalFileName; sl@0: return; sl@0: } sl@0: TChar ellipsis(0x2026); sl@0: --maxWidthInChars; // since the ellipsis will be the first char in aAbbreviatedFileName sl@0: aAbbreviatedFileName.Zero(); sl@0: aAbbreviatedFileName.Append(ellipsis); sl@0: aAbbreviatedFileName.Append(aOriginalFileName.Mid((aOriginalFileName.Length() - 1) - maxWidthInChars + 1, maxWidthInChars)); sl@0: } sl@0: sl@0: sl@0: EXPORT_C TBool BaflUtils::UidTypeMatches(const TUidType& aFileUid, const TUidType& aMatchUid) sl@0: /** Tests whether two UID types match. sl@0: sl@0: A match is made if each UID in aMatchUid is either identical to the corresponding sl@0: one in aFileUid, or is KNullUid. sl@0: sl@0: @param aFileUid The UID type to match sl@0: @param aMatchUid The UID type to match against sl@0: @return ETrue if the UIDs match, otherwise EFalse */ sl@0: { sl@0: for (TInt i=0; iSet(aFolderNameTypedByUser, NULL, NULL); sl@0: User::LeaveIfError(retcode); sl@0: if (targetParse->DrivePresent() || targetParse->PathPresent()) sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: return KErrBadName; // R_EIK_TBUF_INVALID_FOLDER_NAME; sl@0: } sl@0: sl@0: if (!(aFs.IsValidName(aFolderNameTypedByUser))) sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: return KErrBadName; // R_EIK_TBUF_INVALID_FOLDER_NAME; sl@0: } sl@0: sl@0: sl@0: if ((aCurrentPath.Length() + aFolderNameTypedByUser.Length() + 1) > KMaxFileName) sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: return KErrTooBig; //R_EIK_TBUF_FOLDERNAME_TOO_LONG; sl@0: } sl@0: sl@0: sl@0: //TFileName newFolderFullName = aCurrentPath; sl@0: aNewFolderFullName = aCurrentPath; sl@0: if ((aNewFolderFullName.Length() + aFolderNameTypedByUser.Length() + 1) <= KMaxFileName) sl@0: { sl@0: aNewFolderFullName.Append(aFolderNameTypedByUser); sl@0: aNewFolderFullName.Append(KPathDelimiter); sl@0: } sl@0: else sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: return KErrOverflow; sl@0: } sl@0: sl@0: retcode = targetParse->Set(aNewFolderFullName, NULL, NULL); sl@0: if (retcode != KErrNone) sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: return KErrBadName; // R_EIK_TBUF_INVALID_FOLDER_NAME; sl@0: } sl@0: sl@0: CleanupStack::PopAndDestroy(); sl@0: sl@0: return(KErrNone); sl@0: } sl@0: sl@0: void BaflUtils::DoCopyFileL(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch) sl@0: { sl@0: CFileMan* fileMan=CFileMan::NewL(aFs); sl@0: CleanupStack::PushL(fileMan); sl@0: User::LeaveIfError(fileMan->Copy(aSourceFullName,aTargetFullName,aSwitch)); sl@0: CleanupStack::PopAndDestroy(); // fileMan sl@0: } sl@0: sl@0: sl@0: EXPORT_C TInt BaflUtils::CopyFile(RFs& aFs, const TDesC& aSourceFullName, const TDesC& aTargetFullName, TUint aSwitch) sl@0: /** Copies one or more files. sl@0: sl@0: For more details, sl@0: @see CFileMan::Copy() sl@0: @since 5.1 sl@0: @param aFs File server session sl@0: @param aSourceFullName Path indicating the file(s) to be copied. Any path sl@0: components that are not specified here will be taken from the session path. sl@0: @param aTargetFullName Path indicating the directory into which the file(s) sl@0: are to be copied sl@0: @param aSwitch=CFileMan::EOverWrite Set this to zero for no overwriting and sl@0: no recursion; CFileMan::EOverWrite to overwrite files with the same name; or sl@0: CFileMan::ERecurse for recursion. sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes.*/ sl@0: { sl@0: TRAPD(err,DoCopyFileL(aFs,aSourceFullName,aTargetFullName,aSwitch)); sl@0: return err; sl@0: } sl@0: sl@0: void BaflUtils::DoDeleteFileL(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch) sl@0: { sl@0: CFileMan* fileMan=CFileMan::NewL(aFs); sl@0: CleanupStack::PushL(fileMan); sl@0: User::LeaveIfError(fileMan->Delete(aSourceFullName,aSwitch)); sl@0: CleanupStack::PopAndDestroy(); // fileMan sl@0: } sl@0: sl@0: sl@0: EXPORT_C TInt BaflUtils::DeleteFile(RFs& aFs, const TDesC& aSourceFullName, TUint aSwitch) sl@0: /** Deletes one or more files. sl@0: sl@0: For more details, sl@0: @see CFileMan::Delete(). sl@0: @since 5.1 sl@0: @param aFs File server session sl@0: @param aSourceFullName Path indicating the file(s) to be deleted. May either sl@0: be a full path, or relative to the session path. Use wildcards to specify sl@0: more than one file. sl@0: @param aSwitch=0 Specify CFileMan::ERecurse for recursion, sl@0: zero for no recursion. sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes. */ sl@0: { sl@0: TRAPD(err,DoDeleteFileL(aFs,aSourceFullName,aSwitch)); sl@0: return err; sl@0: } sl@0: sl@0: void BaflUtils::DoRenameFileL(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch) sl@0: { sl@0: CFileMan* fileMan=CFileMan::NewL(aFs); sl@0: CleanupStack::PushL(fileMan); sl@0: User::LeaveIfError(fileMan->Rename(aOldFullName,aNewFullName,aSwitch)); sl@0: CleanupStack::PopAndDestroy(); // fileMan sl@0: } sl@0: sl@0: sl@0: EXPORT_C TInt BaflUtils::RenameFile(RFs& aFs, const TDesC& aOldFullName, const TDesC& aNewFullName, TUint aSwitch) sl@0: /** Renames or moves one or more files or directories. sl@0: sl@0: It can be used to move one or more files by specifying different sl@0: destination and source directories. sl@0: For more details, sl@0: @see CFileMan::Rename(). sl@0: @since 5.1 sl@0: @param aFs File server session sl@0: @param aOldFullName Path specifying the file(s) to be renamed. sl@0: @param aNewFullName Path specifying the new name for the files and/or the sl@0: new directory. Any directories specified in this path that do not exist will sl@0: be created. sl@0: @param aSwitch=CFileMan::EOverWrite Specify zero for no overwriting, or sl@0: CFileMan::EOverWrite to overwrite files with the same name. This sl@0: function cannot operate recursively. sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes. */ sl@0: { sl@0: TRAPD(err,DoRenameFileL(aFs,aOldFullName,aNewFullName,aSwitch)); sl@0: return err; sl@0: } sl@0: sl@0: sl@0: EXPORT_C TInt BaflUtils::CheckWhetherFullNameRefersToFolder(const TDesC& aFullName, TBool& aIsFolder) sl@0: /** Checks if a string is a valid folder name. sl@0: sl@0: @param aFullName String to check sl@0: @param aIsFolder ETrue if aFullName is a valid folder name, otherwise EFalse sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes (probably because aFullName cannot be parsed). */ sl@0: { sl@0: aIsFolder = EFalse; sl@0: TInt retcode = BaflUtils::Parse(aFullName); sl@0: if (retcode != KErrNone) sl@0: return(retcode); sl@0: TParsePtrC parse(aFullName); sl@0: if (! parse.NameOrExtPresent()) sl@0: aIsFolder = ETrue; sl@0: return(KErrNone); sl@0: } sl@0: sl@0: EXPORT_C TInt BaflUtils::MostSignificantPartOfFullName(const TDesC& aFullName, TFileName& aMostSignificantPart) sl@0: /** Gets the folder name if the specified item is a valid folder name, otherwise sl@0: gets the file name. sl@0: sl@0: @param aFullName Item to parse sl@0: @param aMostSignificantPart Folder or file name sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes */ sl@0: { sl@0: TBool entryIsAFolder; sl@0: TInt retcode = CheckWhetherFullNameRefersToFolder(aFullName, entryIsAFolder); sl@0: if (retcode != KErrNone) sl@0: return(retcode); sl@0: if (entryIsAFolder) sl@0: { sl@0: aMostSignificantPart = FolderNameFromFullName(aFullName); sl@0: return (KErrNone); sl@0: } sl@0: // assume aFullName refers to a file sl@0: TParsePtrC parse(aFullName); sl@0: aMostSignificantPart = parse.NameAndExt(); sl@0: return(KErrNone); sl@0: } sl@0: sl@0: EXPORT_C TInt BaflUtils::CheckFolder(RFs& aFs, const TDesC& aFolderName) sl@0: /** Checks that the specified folder can be opened. sl@0: sl@0: @param aFs File server session sl@0: @param aFolderName Folder to check sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes */ sl@0: { sl@0: RDir dir; sl@0: TInt retcode = dir.Open(aFs, aFolderName, 0); sl@0: if (retcode == KErrNone) sl@0: dir.Close(); sl@0: return (retcode); sl@0: } sl@0: sl@0: /** sl@0: Checks if the specified drive is read-only. sl@0: Checks that the KMediaAttWriteProtected and EMediaRom flags are both set. sl@0: sl@0: @param aFs File server session sl@0: @param aFullName File name, including drive sl@0: @param aIsReadOnly On return, ETrue if the drive is read-only, otherwise EFalse sl@0: @return KErrNone if successful, otherwise another of the system-wide error codes sl@0: @see BaflUtils::DriveIsReadOnlyInternal sl@0: */ sl@0: EXPORT_C TInt BaflUtils::DiskIsReadOnly(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnly) sl@0: { sl@0: TInt retcode=BaflUtils::Parse(aFullName); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: TParsePtrC parse(aFullName); sl@0: if (!parse.DrivePresent()) sl@0: return KErrBadName; sl@0: TBuf<1> drive=parse.Drive().Left(1); sl@0: TChar driveLetter=drive[0]; sl@0: TInt driveId=0; sl@0: retcode=RFs::CharToDrive(driveLetter,driveId); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: TVolumeInfo volInfo; sl@0: retcode=aFs.Volume(volInfo,driveId); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: aIsReadOnly=(volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected || volInfo.iDrive.iType==EMediaRom); sl@0: return KErrNone; sl@0: } sl@0: sl@0: /** sl@0: Checks if the specified drive is read-only and is an internal drive i.e. non-removable. sl@0: Checks that the KMediaAttWriteProtected and KDriveAttInternal flags are both set. sl@0: sl@0: @param aFs File server session sl@0: @param aFullName File name, including drive sl@0: @param aIsReadOnlyInternal On return, ETrue if the drive is read-only and internal, otherwise EFalse sl@0: @return KErrNone if successful, otherwise another of the system-wide errors codes sl@0: */ sl@0: EXPORT_C TInt BaflUtils::DriveIsReadOnlyInternal(RFs& aFs, const TDesC& aFullName, TBool& aIsReadOnlyInternal) sl@0: { sl@0: TInt retcode=BaflUtils::Parse(aFullName); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: TParsePtrC parse(aFullName); sl@0: if (!parse.DrivePresent()) sl@0: return KErrBadName; sl@0: TBuf<1> drive=parse.Drive().Left(1); sl@0: TChar driveLetter=drive[0]; sl@0: TInt driveId=0; sl@0: retcode=RFs::CharToDrive(driveLetter,driveId); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: TVolumeInfo volInfo; sl@0: retcode=aFs.Volume(volInfo,driveId); sl@0: if (retcode!=KErrNone) sl@0: return retcode; sl@0: aIsReadOnlyInternal=((volInfo.iDrive.iMediaAtt&KMediaAttWriteProtected)&&(volInfo.iDrive.iDriveAtt&KDriveAttInternal)); sl@0: return KErrNone; sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::GetDiskListL(const RFs& aFs,CDesCArray& aArray) sl@0: /** Retrieves a list of all drives on the system. sl@0: sl@0: The file server is interrogated for a list of the drive letters for all available sl@0: drives. sl@0: sl@0: On emulator: sl@0: The removable media is represented by drive X: . sl@0: sl@0: On hardware: sl@0: The removable media is represented by drives D: E: F: and G: . sl@0: sl@0: @param aFs A connected session with the file server. sl@0: @param aArray On return, contains the drive letters that correspond to the sl@0: available drives. The drive letters are uppercase and are in alphabetical sl@0: order. */ sl@0: { // static sl@0: aArray.Reset(); sl@0: TDriveList driveList; sl@0: User::LeaveIfError(aFs.DriveList(driveList)); sl@0: for (TInt ii=0;ii buf; sl@0: buf.Append(drive); sl@0: aArray.AppendL(buf); sl@0: } sl@0: } sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::UpdateDiskListL(const RFs& aFs,CDesCArray& aArray,TBool aIncludeRom,TDriveNumber aDriveNumber) sl@0: /** Retrieves a list of all drives present on the system. sl@0: sl@0: The file server is interrogated for a list of the drive letters for all available sl@0: drives. The drive letter that corresponds to aDriveNumber is added to the sl@0: list regardless of whether it is present, or is corrupt. Also, the C: drive sl@0: is forced onto the list, even if corrupt or not present. sl@0: sl@0: On emulator: sl@0: The removable media is represented by drive X: and is forced onto the list sl@0: unless removed (F5,F4). sl@0: sl@0: On hardware: sl@0: The removable media is represented by drives D: E: F: and G: and is forced sl@0: onto the list regardless of whether it is present, or is corrupt. sl@0: sl@0: @param aFs A connected session with the file server. sl@0: @param aArray On return, contains the drive letters that correspond to the sl@0: available drives. The drive letters are uppercase and are in alphabetical sl@0: order. sl@0: @param aIncludeRom Specify ETrue if the ROM drive should be included in the sl@0: list, EFalse if not. sl@0: @param aDriveNumber The drive to force into the list, e.g. the drive in the sl@0: default path. */ sl@0: { // static sl@0: aArray.Reset(); sl@0: TDriveList driveList; sl@0: User::LeaveIfError(aFs.DriveList(driveList)); sl@0: for (TInt ii=0;ii buf; sl@0: buf.Append(drive); sl@0: aArray.AppendL(buf); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: EXPORT_C TBool BaflUtils::IsFirstDriveForSocket(TDriveUnit aDriveUnit) sl@0: /** Tests whether the specified drive corresponds to the primary partition in a sl@0: removable media slot. sl@0: sl@0: The function assumes that the D: drive corresponds to the primary partition sl@0: on socket 0, and that the E: drive corresponds to the primary partition on sl@0: socket 1 (a socket is a slot for removable media). This mapping may not always sl@0: be the case because it is set up in the variant layer of the Symbian OS. sl@0: sl@0: This function assumes that the drive mappings are contiguous, starting sl@0: from drive D: . sl@0: sl@0: On emulator: sl@0: The removable media is represented by drive X: only. sl@0: sl@0: @param aDriveUnit The drive to check. sl@0: @return ETrue if the drive is the primary partition in a removable media slot. sl@0: ETrue is also returned if the drive is A, B or C:. EFalse is returned otherwise. */ sl@0: { // static sl@0: TDriveInfoV1Buf buf; sl@0: UserHal::DriveInfo(buf); sl@0: sl@0: #ifdef __EPOC32__ sl@0: return ((aDriveUnit-EDriveC)<=buf().iTotalSockets); sl@0: #else // emulator sl@0: return (aDriveUnit==EDriveX || (aDriveUnit-EDriveC)<=buf().iTotalSockets); sl@0: #endif sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::RemoveSystemDirectory(CDir& aDir) sl@0: /** Removes "System" from a list of directory entries. sl@0: sl@0: @param aDir Array of directory entries. */ sl@0: { // static sl@0: STATIC_CAST(BaflDir&,aDir).RemoveSystem(); sl@0: } sl@0: sl@0: EXPORT_C TInt BaflUtils::SortByTable(CDir& aDir,CBaflFileSortTable* aTable) sl@0: /** Sorts files by UID. sl@0: sl@0: The caller supplies a table which specifies the order in which files are to sl@0: be sorted. The files whose UID3 is the first UID in the table appear first. sl@0: The files whose UID3 is the UID specified second appear next, and so on. Files sl@0: whose UID3 is not specified in the table, and directories, appear at the end sl@0: of the list, with directories preceding the files, and with files sorted in sl@0: ascending order of UID3. sl@0: sl@0: This function is used for customising how lists of application files are sorted. sl@0: sl@0: @param aDir The array of files and directories to sort. sl@0: @param aTable A sort order table containing the UIDs to use in the sort. sl@0: @return KErrNone if successful, otherwise one of the standard error codes. */ sl@0: { // static sl@0: return STATIC_CAST(BaflDir&,aDir).SortByTable(aTable); sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::GetDowngradePathL(const RFs& aFs, const TLanguage aCurrentLanguage, RArray& aLanguageArray) sl@0: /** Gets the full language downgrade path for a particular locale. sl@0: sl@0: @param aFs A connected session with the file server. sl@0: @param aCurrentLanguage The language of the locale for which the language downgrade sl@0: path is required. This language will always be returned as the first language sl@0: in aLanguageArray. To get the downgrade path for the language of the current sl@0: locale, specify the language returned by User::Language(). sl@0: @param aLanguageArray On return, contains the language downgrade path. sl@0: @see BaflUtils::NearestLanguageFile() */ sl@0: { sl@0: TLocale currentLocale; sl@0: TNearestLanguageFileFinder languageDowngradePath(aFs); sl@0: TLanguage idealLanguage=IdealLanguage(); sl@0: MakeLanguageDowngradePath(languageDowngradePath.iPath,aCurrentLanguage,idealLanguage, currentLocale); sl@0: aLanguageArray.Reset(); sl@0: const TLanguage* p=languageDowngradePath.iPath; sl@0: while (*p != ELangNone) sl@0: { sl@0: User::LeaveIfError(aLanguageArray.Append(*p)); sl@0: ++p; sl@0: } sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::PersistLocale() sl@0: /** Saves the locale settings in TLocale and the currency symbol to file. sl@0: @deprecated 9.1 sl@0: Persistence and initialisation of system locale data will be performed sl@0: transparently by a separate executable (InilialiseLocale.exe) wich should sl@0: be loaded as part of the system start procedure. sl@0: */ sl@0: { sl@0: // Replaced by new repository based locale initialisation mechanism. sl@0: } sl@0: sl@0: sl@0: sl@0: EXPORT_C TInt BaflUtils::PersistHAL() sl@0: /** Saves the HAL settings to file. sl@0: This will start a new executable and saves HAL attributes to a file, sl@0: little delay because of the creation of new process sl@0: @return KErrNone if suceessful, otheriwse system wide error code. sl@0: */ sl@0: { sl@0: RProcess process; sl@0: _LIT(KHALSettings, "HALSettings.exe"); sl@0: _LIT(KCommandLine, "PERSIST"); sl@0: TInt result = process.Create(KHALSettings, KCommandLine); sl@0: if(result != KErrNone ) sl@0: return result; sl@0: TRequestStatus status; sl@0: process.Logon(status); sl@0: if ( status != KRequestPending) sl@0: { sl@0: process.Kill(0); // abort sl@0: } sl@0: else sl@0: { sl@0: process.Resume(); // logon OK sl@0: } sl@0: User::WaitForRequest(status); sl@0: sl@0: // we can't use the 'exit reason' if the exe panicked as this sl@0: // is the panic 'reason' and may be '0' which cannot be distinguished sl@0: // from KErrNone sl@0: result = process.ExitType() == EExitPanic ? KErrGeneral : status.Int(); sl@0: process.Close(); sl@0: return result; sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::PersistScreenCalibration(const TDigitizerCalibration& aScreenCalibration) sl@0: { sl@0: sl@0: RFs fs; sl@0: TInt err = fs.Connect(); sl@0: if (err == KErrNone) sl@0: { sl@0: // Setting up drive to store Screen data sl@0: TDriveUnit systemDrive(static_cast(RFs::GetSystemDrive())); sl@0: TBuf ScreenFileNameWithDriveAndPath(systemDrive.Name()); sl@0: ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFolder); sl@0: sl@0: // Ensure directory \System\Data exists in target drive sl@0: TRAP(err, EnsurePathExistsL(fs, ScreenFileNameWithDriveAndPath)); sl@0: if(err == KErrNone) sl@0: { sl@0: ScreenFileNameWithDriveAndPath.Append(KScreenCalibrationFileName); sl@0: sl@0: RFile file; sl@0: err = file.Replace(fs,ScreenFileNameWithDriveAndPath,EFileWrite|EFileShareExclusive); sl@0: if (err == KErrNone) sl@0: { sl@0: // Write aScreenCalibration to file. sl@0: TPtrC8 calptr((const TUint8*)&aScreenCalibration, sizeof(TDigitizerCalibration)); sl@0: err = file.Write(calptr); sl@0: } sl@0: file.Close(); sl@0: } sl@0: } sl@0: fs.Close(); sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::InitialiseScreenCalibration(RFs& aFs) sl@0: { sl@0: TFindFile ff(aFs); sl@0: if (ff.FindByDir(KScreenCalibrationFileName, KScreenCalibrationFolder)==KErrNone) sl@0: { sl@0: RFile file; sl@0: if (file.Open(aFs,ff.File(),EFileRead) == KErrNone ) sl@0: { sl@0: TDigitizerCalibration calibrationSetting; sl@0: TPtr8 scrcal((TUint8*)&calibrationSetting, sizeof(TDigitizerCalibration)); sl@0: if( file.Read(scrcal, sizeof( TDigitizerCalibration )) == KErrNone ) sl@0: UserHal::SetXYInputCalibration(calibrationSetting); sl@0: sl@0: } sl@0: file.Close(); sl@0: } sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::InitialiseHAL(RFs&) sl@0: /** Initialise the HAL settings from. sl@0: @deprecated 9.1 sl@0: This function is empty sl@0: */ sl@0: { sl@0: } sl@0: sl@0: EXPORT_C void BaflUtils::InitialiseLocale(RFs& /* aFs */) sl@0: { sl@0: // Replaced by new repository based locale initialisation mechanism. sl@0: } sl@0: sl@0: sl@0: // sl@0: // class CEikFileSortTable sl@0: // sl@0: sl@0: /** sl@0: */ sl@0: EXPORT_C CBaflFileSortTable::CBaflFileSortTable() sl@0: : CArrayFixFlat(EArrayGranularity) sl@0: {} sl@0: sl@0: /** sl@0: * Loads the CBaflFileSortTable using the UIDs read from the TResourceReader supplied in aReader. sl@0: * @param aReader TResourceReader from which UIDS are read. sl@0: * @leave KErrNoMemory if there is insufficient memory available or one of the system wide error codes. sl@0: */ sl@0: EXPORT_C void CBaflFileSortTable::ConstructFromResourceL(TResourceReader& aReader) sl@0: { sl@0: const TInt count=aReader.ReadInt16(); sl@0: for (TInt i=0;iFind(entry,key,index)==KErrNone) sl@0: { sl@0: entry=(*iArray)[index]; sl@0: if (entry.IsDir()) sl@0: iArray->Delete(index); sl@0: } sl@0: }; sl@0: sl@0: /* sl@0: This function gets the element at position "aPos" of aEntries array and inserts sl@0: it to poition "aNewPos". The element size is "aSize". After the operation the array size sl@0: grows by 1, the element at "aPos" position moves one position forward. sl@0: This function must be called only from BaflDir::SortByTable() and the insert position sl@0: is always less or equal than the position of the element to be inserted. sl@0: */ sl@0: static void InsertL(CArrayPakFlat* aEntries, TInt aPos, TInt aNewPos, TInt aSize) sl@0: { sl@0: __ASSERT_DEBUG(aPos >= aNewPos, User::Invariant()); sl@0: //Expand the array adding one empty entry at "aNewPos" position. sl@0: aEntries->ExpandL(aNewPos, aSize); sl@0: //After successfull "ExpandL" call "aPos" must be incremented by 1. sl@0: //Copy the entry from "aPos + 1" position to "aNewPos" position sl@0: (*aEntries)[aNewPos] = (*aEntries)[++aPos]; sl@0: } sl@0: sl@0: /** sl@0: Copied from f32file.inl (now moved to f32file_private.h) sl@0: Returns the minimum uncompressed size of the TEntry object, including the valid sl@0: portion of the name string. The returned value is aligned to 4-byte boundary and sl@0: length includes private members. sl@0: sl@0: @internalTechnology sl@0: @return minimum uncompressed size of TEntry object sl@0: */ sl@0: sl@0: TInt BaflDir::MinEntrySize(const TEntry & aEntry) sl@0: { sl@0: return(sizeof(TUint)+sizeof(TInt)+sizeof(TTime)+sizeof(TInt)+sizeof(TUidType)+ sl@0: Align4(aEntry.iName.Size()) + 2*sizeof(TInt)); sl@0: } sl@0: sl@0: TInt BaflDir::SortByTable(CBaflFileSortTable* aTable) sl@0: /** sl@0: Sort into order from given table. sl@0: Any file with uid[2] matching an entry in the table will be sorted relative to sl@0: others in the table and before any files with no matching uid. sl@0: For Example: Assume UID table is filled with below 2 UID's sl@0: table[0] =0x10003a64 and table[1] =0x10003a5c. Then file with UID[2]=0x10003a64 sl@0: will be sorted first in the list followed by file with UID[2]=0x10003a5c. Rest sl@0: files will be sorted in the ascending order of UID[2] with directories preceding sl@0: the files. sl@0: sl@0: @param aTable A sort order table containing the UIDs to use in the sort. sl@0: @return KErrNone if suceessful, otheriwse another system-wide error code. sl@0: */ sl@0: { sl@0: TInt r=this->Sort(EDirsFirst|ESortByUid); sl@0: if(r != KErrNone) sl@0: { sl@0: return r; sl@0: } sl@0: const TInt tableCount=aTable->Count(); sl@0: const TInt count=iArray->Count(); sl@0: TInt sortedInsertionPoint = 0; sl@0: for (TInt i=0;iAt(i); sl@0: for (TInt j=sortedInsertionPoint;jiType[2]) sl@0: { sl@0: TRAPD(insertErr, ::InsertL(iArray, j, sortedInsertionPoint++, MinEntrySize(*pEntry))); sl@0: if(insertErr!=KErrNone) sl@0: { sl@0: return insertErr; sl@0: } sl@0: iArray->Delete(j+1); sl@0: } sl@0: } sl@0: } sl@0: iArray->Compress(); sl@0: return KErrNone; sl@0: }