sl@0: /* sl@0: * Copyright (c) 2002-2008 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: Provides formatting (grouping) for plain phone numbers sl@0: * sl@0: * sl@0: */ sl@0: sl@0: sl@0: #ifndef C_NUMBER_GROUPING_H sl@0: #define C_NUMBER_GROUPING_H sl@0: sl@0: // #define __TEST_AS_EXE__ // put this in if you want all the test code to run and swap the mmps... sl@0: sl@0: #include "NumberGroupingStd.h" sl@0: sl@0: class TResourceReader; sl@0: class CRegularExpression; sl@0: class CPNGNumberGroupingExtension; sl@0: sl@0: /** sl@0: * Engine class to format plain phone numbers by inserting grouping separators. sl@0: * Both unformatted and formatted buffers are maintained by this class. sl@0: * A reversed formatted buffer is also available to assist the client in performing wrapping. sl@0: * sl@0: * @lib NumberGrouping.lib sl@0: */ sl@0: NONSHARABLE_CLASS(CPNGNumberGrouping) : public CBase sl@0: { sl@0: sl@0: /** sl@0: * Central Repository key values for KNumberGrouping key in NumberGroupingCRKeys.h sl@0: */ sl@0: public: sl@0: enum TNumberGroupingCRValues sl@0: { sl@0: /** Number grouping disabled */ sl@0: ENumberGroupingDisabled = 0, sl@0: /** Number grouping enabled (USA) */ sl@0: ENumberGroupingEnabled = 1 sl@0: }; sl@0: sl@0: public: sl@0: IMPORT_C static CPNGNumberGrouping* NewL(TInt aMaxLength = 0, TBool aReversed = EFalse); sl@0: IMPORT_C static CPNGNumberGrouping* NewLC(TInt aMaxLength = 0, TBool aReversed = EFalse); sl@0: IMPORT_C ~CPNGNumberGrouping(); sl@0: sl@0: IMPORT_C TInt Insert(TInt aIndex, TText aChar); sl@0: IMPORT_C TInt Delete(TInt aIndex); sl@0: IMPORT_C TInt Append(TText aChar); sl@0: sl@0: /** sl@0: * Sets the new unformatted phone number. sl@0: * sl@0: * Formatting does not actually occur until an API is called that accesses in some way the formatted buffer sl@0: * or one of its characteristics sl@0: * sl@0: * @param aNumber Ungrouped phone number to be copied into the NumberGrouping engine's unformatted buffer sl@0: * @return KErrOverflow if the number is too long for the length of the unformatted buffer sl@0: */ sl@0: IMPORT_C TInt Set(const TDesC& aNumber); sl@0: sl@0: /** sl@0: * @return Length of the currently formatted (grouped) buffer sl@0: */ sl@0: IMPORT_C TInt Length() const; sl@0: sl@0: /** sl@0: * @return Length of the currently unformatted buffer sl@0: */ sl@0: IMPORT_C TInt UnFormattedLength() const; sl@0: sl@0: /** sl@0: * This returns the maximum size of the unformatted buffer. This is the value that was provided during sl@0: * construction. sl@0: * sl@0: * Descriptors provided to Set() must be shorter than this length. sl@0: * sl@0: * @return maximum length of the unformatted buffer. sl@0: */ sl@0: IMPORT_C TInt MaxDisplayLength() const; sl@0: sl@0: /** sl@0: * Routine to determine if the character at a given position in the formatted phone number sl@0: * is a space character derived from the number grouping. That is, not part of the supplied phone number sl@0: * proper, but a space character from the number grouping. sl@0: * sl@0: * A client can check the descriptor returned by FormattedNumber() directly to perform a simple test sl@0: * of whether the character is a Space or not (whether or not derived from number grouping formatting) sl@0: * sl@0: * Note also that this returns EFalse if the character is some other formatting character besides sl@0: * space. To determine that, use IsCharacterInsertedByNumberGrouping(). sl@0: * sl@0: * @param aPos The index of the character of interest sl@0: * @return EFalse iff the characer at aPos in the formatted buffer is a space coming from grouping sl@0: */ sl@0: IMPORT_C TBool IsSpace(TInt aPos) const; sl@0: sl@0: /** sl@0: * Access to section of the formatted buffer. sl@0: * This routine returns a descriptor for the indicated range in the formatted (grouped) internal sl@0: * buffer. If there are spaces at either end of the indicated section of the formatted buffer, then sl@0: * the returned descriptor is adjusted to point to the trimmed buffer in order to avoid the spaces. sl@0: * sl@0: * @param aFrom Inclusive starting index sl@0: * @param aTo Inclusive end index sl@0: * @return reference to const non-modifiable descriptor for the indicated, trimmed text sl@0: */ sl@0: IMPORT_C const TDesC& FormattedNumber(TInt aFrom, TInt aTo) const; sl@0: sl@0: IMPORT_C const TDesC& FormattedNumber() const; sl@0: sl@0: /** sl@0: * Access to part of the reverse formatted number. If there are spaces at either end of the indicated sl@0: * section of the formatted buffer, then the returned descriptor is adjusted to point to the trimmed sl@0: * buffer in order to avoid the spaces. sl@0: * sl@0: * Returns KNullDesC if the feature has not been enabled by passing ETrue to the sl@0: * parameter aReversed during construction sl@0: * sl@0: * @param aFrom lower (inclusive) limit of the text to look at sl@0: * @param aTo upper (inclusive) limit of the text to look at sl@0: * @return Reference to descriptor containing the selected text sl@0: */ sl@0: IMPORT_C const TDesC& ReverseFormattedNumber(TInt aFrom, TInt aTo) const; sl@0: sl@0: /** sl@0: * Access to the reverse formatted number sl@0: * sl@0: * Returns KNullDesC if the feature has not been enabled by passing ETrue to the sl@0: * parameter aReversed during construction sl@0: * sl@0: * @return Reference to descriptor containing the reverse formatted text sl@0: */ sl@0: IMPORT_C const TDesC& ReverseFormattedNumber() const; sl@0: IMPORT_C const TDesC& Selection(TInt aFrom, TInt aTo) const; sl@0: sl@0: IMPORT_C const TDesC& UnFormattedNumber(TInt aFrom, TInt aTo) const; sl@0: IMPORT_C const TDesC& UnFormattedNumber() const; sl@0: sl@0: /** sl@0: * This method allows the client to determine if the indexed character is a number grouping supplied sl@0: * character. Specifically, this means that this character originates in the number grouping formatting sl@0: * and not from the supplied unformatted phone number. sl@0: * sl@0: * Where the number has not been grouped (e.g. because there is an invalid phone number character in the sl@0: * supplied descriptor), this method returns EFalse, even if the character pointed to may be of the nature sl@0: * of a number grouping character. Use IsChangedByGrouping() to see if the number has been changed by sl@0: * grouping at all. sl@0: * sl@0: * Where a client is interested purely in the nature of the characters rather than whether they come from sl@0: * grouping or not, he may examine the examine the text via the descriptor reference returned by sl@0: * FormattedNumber(). sl@0: * sl@0: * @since Series 60 2.6 sl@0: * @param aPos The index provided is for the formatted number. sl@0: * @return EFalse iff the character at the supplied index is part of the supplied phone number sl@0: */ sl@0: IMPORT_C TBool IsCharacterInsertedByNumberGrouping(TInt aPos) const; sl@0: sl@0: /** sl@0: * Method to determine if the current number has been changed by number grouping. sl@0: * If this returns EFalse, then FormattedNumber() and UnFormattedNumber() refer to descriptors of with identical sl@0: * content. sl@0: * If this method returns ETrue, then the descriptors that would be returned by the two accessor APIs refer sl@0: * to descriptors with different content. sl@0: * sl@0: * @since Series 60 2.6 sl@0: * @return ETrue iff formatting of the number has made an effective difference. sl@0: */ sl@0: IMPORT_C TBool IsChangedByGrouping() const; sl@0: sl@0: /** sl@0: * @return return iLanguage. sl@0: */ sl@0: inline TLanguage Language() const; sl@0: sl@0: sl@0: public: sl@0: sl@0: TLanguage iForceLanguage; sl@0: sl@0: private: // private classes and enums sl@0: sl@0: class TPNGSeparator sl@0: { sl@0: public: sl@0: TPNGSeparator(); sl@0: TPNGSeparator( TInt aPos, TText aSeparatorCharacter ); sl@0: public: sl@0: TInt iPosition; sl@0: TText iSeparatorCharacter; sl@0: }; sl@0: sl@0: class TPNGGroupingInfo sl@0: { sl@0: public: sl@0: TPNGGroupingInfo(); sl@0: public: sl@0: TInt iMinNumberOfDigits; sl@0: TInt iMaxNumberOfDigits; sl@0: RArray iAfterPositions; // Positions of separators "after" the beginning sl@0: TPNGSeparator iBeforePosition; // Positions of separators "before" the end sl@0: }; sl@0: sl@0: // Constant for no pattern in use: sl@0: enum sl@0: { sl@0: ENoMatchedPattern = -1 sl@0: }; sl@0: sl@0: private: // Constructors sl@0: CPNGNumberGrouping( TInt aMaxLength, TBool aReserved); sl@0: void ConstructL(); sl@0: sl@0: private: sl@0: TLanguage doReadLanguageFromSharedData() const; sl@0: sl@0: void doReadFormatInfoFromResourceFileL(); sl@0: void doClearGroupingItemsList(); sl@0: void doClearFormattedNumbers(); sl@0: sl@0: void doNumberGroupingL() const; sl@0: void doNumberSquashing() const; sl@0: sl@0: /** sl@0: * Read and process a single NUMBER_GROUPING_ITEM resource structure sl@0: */ sl@0: void ReadGroupingSchemeL( sl@0: TResourceReader& aResourceReader, sl@0: RPointerArray& aGroupingPatternsList, sl@0: TInt& aMaxExtraCharacters ); sl@0: sl@0: /** sl@0: * Read and skip a single NUMBER_GROUPING_ITEM resource structure sl@0: */ sl@0: void SkipGroupingSchemeL( TResourceReader& aResourceReader ) const; sl@0: sl@0: /** sl@0: * Process the format pattern for "after positions" sl@0: */ sl@0: void ParseForAfterPositions( sl@0: const TDesC& aFormatPattern, sl@0: TPNGGroupingInfo* aGroupingInfo, sl@0: const TDesC& aWildcardedMatchingPattern, sl@0: TInt& aMaxExtraCharacters, sl@0: TBool& trailingPossible ) const; sl@0: sl@0: /** sl@0: * Process the format pattern for a before positions sl@0: */ sl@0: void ParseForBeforePosition( sl@0: const TDesC& aFormatPattern, sl@0: CPNGNumberGrouping::TPNGGroupingInfo* aGroupingInfo, sl@0: TInt& aMaxExtraCharacters sl@0: ) const; sl@0: sl@0: /** sl@0: * This routine is used to find a wildcarded version of the matching pattern sl@0: * provided in the "initialDigits" element for a grouping scheme read from resource. sl@0: * It uses the services of the CRegularExpression class, an instance of which is constructed sl@0: * with the provided aMatchString pattern only. sl@0: * sl@0: * The client must supply a modifiable descriptor long enough to hold the wildcarded version sl@0: * of the pattern. sl@0: * sl@0: * For each character index, if there is only one possible valid character, this puts in that sl@0: * character. If there are more than one, then the supplied aWildcardChar is inserted. The sl@0: * initialDigits element uses a full stop as a numeric wildcard; this is replaced with the nominated sl@0: * wildcard. sl@0: * sl@0: * Rules: (where 'n, has been passed is used as the wildcard) sl@0: * "" -> "" sl@0: * ( e.g. "1" -> "1" ) sl@0: * "+" -> "+" sl@0: * "." -> "n" sl@0: * "[0-3]" -> "n" sl@0: * sl@0: * @param aMatchString Regular expression to provide example of sl@0: * @param aWildcardChar The character to put in the example pattern if there is no single sl@0: * valid character at that point sl@0: * @param aWildcardMatchString Descriptor to write the wildcarded match pattern into sl@0: */ sl@0: void GetWildcardVersionOfMatchStringL( sl@0: const TDesC& aMatchString, sl@0: TText aWildcard, sl@0: TDes& aWildcardMatchString ) const; sl@0: sl@0: /** sl@0: * This method expresses the policy of what characters may form part of a phone number sl@0: * Note that this method is valid even it there is no formatting going on. It is an intrinsic sl@0: * test on the character itself sl@0: */ sl@0: TBool IsValidPhoneNumberCharacter(TText aCharacter) const; sl@0: sl@0: /** sl@0: * Examines the unformatted number, counting how many digits are found before non-digit sl@0: * characters or the end is encountered. sl@0: * Returns 0 if Set() has not been called. sl@0: */ sl@0: TInt LengthToGroup() const; sl@0: sl@0: /** sl@0: * Perform number grouping using pattern at given index. Grouping is only applied to the leading sl@0: * aLengthToGroup characters. sl@0: */ sl@0: void doNumberGroupingForPatternL( TInt aMatchedPattern, TInt aLengthToGroup ) const; sl@0: sl@0: private: // private data sl@0: sl@0: HBufC* iUnformattedNumber; sl@0: mutable TPtrC iUnformattedNumberPtr; sl@0: sl@0: mutable HBufC* iFormattedNumber; sl@0: mutable TPtrC iFormattedNumberPtr; sl@0: sl@0: mutable HBufC* iReverseFormattedNumber; sl@0: mutable TPtrC iReverseFormattedNumberPtr; sl@0: sl@0: mutable TPtrC iSelectionPtr; sl@0: sl@0: mutable TLanguage iLanguage; // the system language sl@0: TInt iMaxUnformattedLength; sl@0: TBool iReversed; sl@0: sl@0: CRegularExpression* iRegExp; // the patterns for matching sl@0: RPointerArray iGroupingItemsList; // the formatting info sl@0: sl@0: mutable TInt iMatchedPatternIndex; // keep track of what pattern is matched sl@0: CPNGNumberGroupingExtension* iExtension; sl@0: }; sl@0: sl@0: inline TLanguage CPNGNumberGrouping::Language() const sl@0: { sl@0: return iLanguage; sl@0: } sl@0: sl@0: sl@0: #endif // C_NUMBER_GROUPING_H sl@0: sl@0: // End of File