uifw/eikctl/src/EIKMFNE.CPP
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 25 May 2010 12:58:19 +0300
branchRCL_3
changeset 12 941195f2d488
parent 9 0aa5fbdfbc30
child 15 c52421ed5f07
permissions -rw-r--r--
Revision: 201019 Kit: 2010121

/*
* Copyright (c) 1997-1999 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "Eclipse Public License v1.0"
* which accompanies this distribution, and is available
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
*
* Initial Contributors:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:
*
*/


#include <basched.h>
#include <barsread.h>
#include <eikenv.h>
#include <eikcmbut.h>
#include <eikmsg.h>
#include <gulcolor.h>
#include <gulutil.h>
#include <eikon.hrh>
#include <eikmfne.pan>
#include <eikmfne.h>
#include <eikctl.rsg>
#include <coecobs.h>
#include <AknUtils.h>
#include <avkon.hrh>
#include <aknenv.h>

// Skins drawing support
#include <AknsDrawUtils.h>
#include <aknappui.h>
#include <AknSettingCache.h>
#include <numberconversion.h>
#include <aknsoundsystem.h>
#include <AknTextDecorationMetrics.h>
#include <AknLayoutFont.h>
#include <AknBidiTextUtils.h>
#include <aknextendedinputcapabilities.h>
#include <fepbase.h>

#include <AknTasHook.h>
#include <eikdialg.h>

#include <touchfeedback.h>

// const TInt KMaxLatitudeDegrees=90;
// const TInt KMaxLongitudeDegrees=180;
const TInt KMaxMinSecValue=59;


namespace EikMfneUtils
    {
    /**
    IsAnyDigit, Is this a digit character any kind of numeral which can be mapped to 0-9?
    @internalComponent
    @return ETrue if this character is a numeral
    @param aCharacter character to be tested 
    */
    LOCAL_D TBool IsAnyDigit(TChar aCharacter)
        {
        return ( aCharacter.GetNumericValue() != KErrNotFound);
        }
    }

enum
    {
    EGapAboveText=0,
    EMfneFieldExtraHeight=2, // This constant is not used in Scalable UI
    EGapBelowText=0,
    EGapLeftOfFirstField=0,
    EGapRightOfLastField=0,
    EExtraHighlightHeight=0,
    };

// _LIT(KSpace," ");
_LIT(KPlus,"+");
_LIT(KMinus,"-");

// global functions

GLDEF_C TInt ExtraBaselineOffset()
    {
    TInt baselineOffset(0);
    return baselineOffset;
    }

// For positioning mfne vertically to the correct position.
GLDEF_C TInt VerticalOffsetForMfnePosition()
    {
    TAknLayoutId layout;
    CAknEnv::Static()->GetCurrentLayoutId( layout );
    return 0;
    }

GLDEF_C void Panic(TEikMfnePanic aPanic)
    {
    _LIT(KPanicCat,"EIKON-MFNE");
    User::Panic(KPanicCat, aPanic);
    }

NONSHARABLE_CLASS(CEikMfneExtension) : public CBase, public MCoeFepAwareTextEditor 
    {
public:
    static CEikMfneExtension* NewL();
    virtual ~CEikMfneExtension();
    
// From MCoeFepAwareTextEditor
public:
    void StartFepInlineEditL( const TDesC &aInitialInlineText,
        TInt aPositionOfInsertionPointInInlineText, TBool aCursorVisibility,
        const MFormCustomDraw *aCustomDraw,
        MFepInlineTextFormatRetriever &aInlineTextFormatRetriever,
        MFepPointerEventHandlerDuringInlineEdit
            &aPointerEventHandlerDuringInlineEdit );

    void UpdateFepInlineTextL( const TDesC &aNewInlineText,
        TInt aPositionOfInsertionPointInInlineText );
    
    void SetInlineEditingCursorVisibilityL( TBool aCursorVisibility );
    void CancelFepInlineEdit();
    TInt DocumentLengthForFep() const;
    TInt DocumentMaximumLengthForFep() const;
    void SetCursorSelectionForFepL( 
        const TCursorSelection &aCursorSelection );
        
    void GetCursorSelectionForFep( TCursorSelection &aCursorSelection ) const;
    void GetEditorContentForFep( TDes &aEditorContent, TInt aDocumentPosition,
        TInt aLengthToRetrieve ) const;
        
    void GetFormatForFep( TCharFormat &aFormat, TInt aDocumentPosition ) const;
    void GetScreenCoordinatesForFepL( TPoint &aLeftSideOfBaseLine,
        TInt &aHeight, TInt &aAscent, TInt aDocumentPosition ) const;

private:
    void DoCommitFepInlineEditL();
    void FieldToSelection( TInt aField, TCursorSelection &aCursorSelection ) const;

private: 
    CEikMfneExtension();
    void ConstructL();

public:
    MAknsControlContext* iExternalSkinControlContext;
    TInt iAknSkinColorIndex;
    TInt iSkinIdForTextBackgroundColor; 
    CAknExtendedInputCapabilities* iExtendedInputCapabilities;
    CEikMfne* iEditor; // not owned
    CAknExtendedInputCapabilities::CAknExtendedInputCapabilitiesProvider*
        iExtendedInputCapabilitiesProvider;
    TInt iClipGcToRect; // Mainly for 0/1 values
    TInt iDisablePenInput; // Mainly for 0/1 values
    TBool iCursorShown; 
    TBool iFingerSupport;
    TInt iFingerParam;
    TBool iHighlightAll; 
    TBool iTouchActivated;
    TCallBack iValidateValueCallBack;
    
    MTouchFeedback* iFeedback;
    TBool iLaunchPenInputAutomatic;
    TBool iPartialScreenInput;
    };

// Implementation of the extension
CEikMfneExtension* CEikMfneExtension::NewL()
    {
    CEikMfneExtension* self = new (ELeave) CEikMfneExtension();
    CleanupStack::PushL( self );
    self->ConstructL();
    CleanupStack::Pop();
    return self;
    }

CEikMfneExtension::~CEikMfneExtension()
    {
    delete iExtendedInputCapabilities;
    delete iExtendedInputCapabilitiesProvider;
    }

CEikMfneExtension::CEikMfneExtension():iAknSkinColorIndex(KErrNotFound), iSkinIdForTextBackgroundColor(KErrNotFound)
    , iFeedback(MTouchFeedback::Instance())
    {
    }

void CEikMfneExtension::ConstructL()
    {
    iExtendedInputCapabilities = CAknExtendedInputCapabilities::NewL();
    iExtendedInputCapabilitiesProvider =
        new (ELeave) CAknExtendedInputCapabilities::CAknExtendedInputCapabilitiesProvider;
        
    iExtendedInputCapabilitiesProvider->SetExtendedInputCapabilities(
        iExtendedInputCapabilities );
    }


void CEikMfneExtension::StartFepInlineEditL( const TDesC &/*aInitialInlineText*/,
        TInt /*aPositionOfInsertionPointInInlineText*/, TBool /*aCursorVisibility*/,
        const MFormCustomDraw */*aCustomDraw*/,
        MFepInlineTextFormatRetriever &/*aInlineTextFormatRetriever*/,
        MFepPointerEventHandlerDuringInlineEdit
            &/*aPointerEventHandlerDuringInlineEdit*/ )
    {
    // No implementation
    }


void CEikMfneExtension::UpdateFepInlineTextL( const TDesC &/*aNewInlineText*/,
        TInt /*aPositionOfInsertionPointInInlineText*/ )
    {
    // No implementation
    }
    
    
void CEikMfneExtension::SetInlineEditingCursorVisibilityL( TBool /*aCursorVisibility*/ )
    {
    // No implementation
    }
    
    
void CEikMfneExtension::CancelFepInlineEdit()
    {
    // No implementation
    }
    
    
TInt CEikMfneExtension::DocumentLengthForFep() const
    {
    TInt documentLength( 0 );
    
    // The fields' concrete Text() function must be
    // properly implemented for this to work.
    for ( TInt i = 0; i < iEditor->NumFields(); ++i )
        {
        documentLength += iEditor->Field( i )->FieldText().Length();
        }

    return documentLength;
    }
    
    
TInt CEikMfneExtension::DocumentMaximumLengthForFep() const
    {
    // We can't reach the field descriptors' MaxLength
    // from here, as the implementations may vary
    return DocumentLengthForFep();
    }
    
    
void CEikMfneExtension::SetCursorSelectionForFepL(
    const TCursorSelection &aCursorSelection )
    {
    TInt documentLength( 0 );
    TInt cursorPos = aCursorSelection.iCursorPos;
    TInt i( 0 );
    // The fields' concrete Text() function must be
    // properly implemented for this to work.
    for ( ; i < iEditor->NumFields(); ++i )
        {
        documentLength += iEditor->Field( i )->FieldText().Length();
  
        if ( cursorPos <= documentLength )
            {
            if ( iEditor->Field( i )->IsEditable() )
                {
                iEditor->HighlightField( i );
                break;
                }
            else
                {
                cursorPos++;
                }
            }
        }
    
    if ( iFingerSupport && iFingerParam == CEikMfne::EnableWithAllHighlight )
        {
        TBool oldHighlightAll( iHighlightAll );                
        iHighlightAll = ( aCursorSelection.Length() > documentLength && 
            i == iEditor->NumFields() );
        iTouchActivated = !iHighlightAll;
        if ( oldHighlightAll != iHighlightAll )
            {
            CEikMfneField* currentField( iEditor->Field( 
                iEditor->CurrentField() ) );
            if ( currentField->HighlightType() == CEikMfneField::ECursor )
                {
                const CFont& font=*( iEditor->Font() );
                TBool dataAltered( EFalse );
                TBool error( EFalse );
                CEikonEnv* env( CEikonEnv::Static() );
                currentField->HandleDeHighlight( font, *env, dataAltered, 
                    error );
                env->HideCursor( iEditor );
                iCursorShown = EFalse;                
                }
            iEditor->DrawDeferred();
            iEditor->ReportUpdate();
            if ( iHighlightAll )
                {
                iEditor->SetFirstEditableField();
                }
            }
        if ( !iTouchActivated )
            {
            iValidateValueCallBack.CallBack();
            iEditor->ReportStateChangeEventL();
            }
        }
    }
        
void CEikMfneExtension::FieldToSelection( TInt aField, TCursorSelection &aCursorSelection ) const
    {
    TInt fieldStart( 0 );
    TInt fieldEnd( 0 );
    TInt fieldIndex( 0 );
    
    while ( fieldIndex < iEditor->NumFields() && fieldIndex <= aField )
        {
        fieldStart = fieldEnd;
        fieldEnd += iEditor->Field( fieldIndex )->FieldText().Length();
        
        if ( fieldIndex == aField )
            {
            aCursorSelection.SetSelection( fieldStart, fieldEnd );
            }
        
        fieldIndex++;
        }
    }
    
void CEikMfneExtension::GetCursorSelectionForFep( TCursorSelection &aCursorSelection ) const
    {
    TInt currentField = iEditor->CurrentField();
    
    // ENullIndex is private
    if ( iFingerSupport && ( iFingerParam == CEikMfne::EnableWithAllHighlight ) && 
        iHighlightAll )
        {
        aCursorSelection.SetSelection( 0, DocumentLengthForFep() );        
        }
    else if ( currentField >= 0 && currentField < iEditor->NumFields() )
        {
        CEikMfneField* field = iEditor->Field( currentField );
        
        FieldToSelection( currentField, aCursorSelection );
    
        // If the field is not highlighted, we'll assume that the
        // cursor is in the end of the field.
        if ( field->HighlightType() == CEikMfneField::ECursor )
            {
            aCursorSelection.iCursorPos = aCursorSelection.iAnchorPos;
            }
        }
    else
        {
        aCursorSelection.SetSelection( -1, -1 );
        }
    }
    
    
void CEikMfneExtension::GetEditorContentForFep( TDes &aEditorContent, TInt aDocumentPosition,
        TInt aLengthToRetrieve ) const
    {
    TInt length = DocumentLengthForFep();
    
    HBufC* content = HBufC::New( length );
    if ( content )
        {
        TPtr contentPtr = content->Des();
        
        for ( TInt i = 0; i < iEditor->NumFields(); ++i )
            {
            contentPtr += iEditor->Field( i )->FieldText();
            }
            
        aEditorContent =
            contentPtr.Mid( aDocumentPosition, aLengthToRetrieve );
            
        delete content;
        }
    }
      
        
void CEikMfneExtension::GetFormatForFep( TCharFormat &/*aFormat*/, TInt /*aDocumentPosition*/ ) const
    {
    // No implementation
    }
    
// FEP uses this function to get the coordinates of bottom of current cursor
void CEikMfneExtension::GetScreenCoordinatesForFepL(
    TPoint &aLeftSideOfBaseLine, TInt &aHeight, TInt &aAscent,
    TInt /*aDocumentPosition*/ ) const
    {
    TInt cursorWidth( 0 );
    iEditor->GetCursorInfo( aLeftSideOfBaseLine, aHeight, cursorWidth, 
        aAscent );
    TPoint screenPos( iEditor->PositionRelativeToScreen() );
    TPoint edwinPos = iEditor->Position();
    aLeftSideOfBaseLine.iY += ( screenPos.iY - edwinPos.iY );    
    }
    
void CEikMfneExtension::DoCommitFepInlineEditL()
    {
    // No implementation
    }

// CEikMfneField

EXPORT_C CEikMfneField::CEikMfneField()
    :iMinimumWidthInPixels(0)
    {
    __DECLARE_NAME(_S("CEikMfneField"));
    }

void CEikMfneField::Draw(CWindowGc& aGc, const CFont& aFont, const TPoint& aTopLeft) const
    {
    // Note that skin background drawing is currently performed in CEikMfne::DrawRange.
    // So this is just responsible for text and (depending on highlighting) the highlight block    
    TInt topMargin(0);
    TInt bottomMargin(0);
    TInt extraHeight(0);
    TAknTextDecorationMetrics metrics( &aFont );
    metrics.GetTopAndBottomMargins( topMargin, bottomMargin );
    extraHeight = topMargin + bottomMargin;
    const CAknLayoutFont* layoutFont = CAknLayoutFont::AsCAknLayoutFontOrNull( &aFont );
    TInt textPaneHeight(0);
    TInt textPaneAscent(0);

    // Get the pane metrics from the best source of information
    if ( layoutFont )
        {
        textPaneHeight = layoutFont->TextPaneHeight();
        textPaneAscent = layoutFont->TextPaneTopToBaseline();
        }
    else
        {
        textPaneHeight = aFont.HeightInPixels();
        textPaneAscent = aFont.AscentInPixels();
        }

    // This is similar to TAknLayoutText's draw, but includes margins.
    // Text and background are draw in one operation. Positioning of the text horizontally
    // w.r.t. aTopMargin is done with the "aLeftMargin" parameter. Horizontal extent is
    // controlled by the width of the "aBox" parameter
    TInt leftMargin(0);
    TInt rightMargin(0);
    metrics.GetLeftAndRightMargins( leftMargin, rightMargin );
    
    // Convert the text to visual
    TInt textWidth    = WidthInPixels( aFont );
    TPtrC drawText    = Text();
    HBufC* visualText = HBufC::New( drawText.Length() + KAknBidiExtraSpacePerLine );
    
    if ( visualText )
        {
        TPtr visualTextPtr = visualText->Des();
                        
        AknBidiTextUtils::ConvertToVisualAndClip(
            drawText,
            visualTextPtr,
            aFont,
            textWidth,
            textWidth,
            AknBidiTextUtils::EImplicit,
            0xFFFF ); // no clipping char
            
        drawText.Set( *visualText );
        }
    
    aGc.DrawText( drawText,
        TRect( aTopLeft, // top left. Should be control top left
        TSize( textWidth, //width
        textPaneHeight + extraHeight )), // total height
        textPaneAscent + topMargin, // distance from top to baseline
        CGraphicsContext::ECenter); // alignment to center
        
    delete visualText;
    }

TInt CEikMfneField::WidthInPixels(const CFont& aFont) const
    {
    return Max( iMinimumWidthInPixels, aFont.TextWidthInPixels(Text()) + AdditionalWidthForHighlights(aFont) );
    }

TInt CEikMfneField::DistanceFromStartOfFieldToEndOfTextInPixels(const CFont& aFont) const
    {
    // Approximation is that the spacing in glyphs is on the right. Therefore the additional spacing will
    // have been put on the left. See the draw routine call in CEikMMfneField::Draw
    TInt textwidth = aFont.TextWidthInPixels(Text()); 
    return  textwidth + AdditionalWidthForHighlights(aFont)+ (WidthInPixels(aFont)-textwidth)/2;
    }

EXPORT_C TBool CEikMfneField::IsEditable() const
    {
    return EFalse;
    }

EXPORT_C TBool CEikMfneField::IsValid() const
    {
    return ETrue;
    }

EXPORT_C CEikMfneField::THighlightType CEikMfneField::HighlightType() const
    {
#if defined(_DEBUG)
    Panic(EEikPanicMfneFieldIsNotEditable1);
#endif
    return EInverseVideo; // dummy return to prevent compiler error
    }

EXPORT_C void CEikMfneField::HandleKey(const CFont&, const TKeyEvent&, TBool, TBool&, TInt&)
    {
#if defined(_DEBUG)
    Panic(EEikPanicMfneFieldIsNotEditable2);
#endif
    }

EXPORT_C void CEikMfneField::HandleDeHighlight(const CFont&, CEikonEnv&, TBool&, TBool&)
    {
#if defined(_DEBUG)
    Panic(EEikPanicMfneFieldIsNotEditable3);
#endif
    }

EXPORT_C void CEikMfneField::HandleLeftOrRightArrow(TChar aKey, TBool& aDataAltered, TInt& aHighlightIncrement)
    {
    __ASSERT_DEBUG((aKey==EKeyLeftArrow) || (aKey==EKeyRightArrow), Panic(EEikPanicMfneArrowKeyExpected));
    aHighlightIncrement=(aKey==EKeyLeftArrow)? -1: 1;
    aDataAltered=ETrue;
    }

TInt CEikMfneField::AdditionalWidthForHighlights(const CFont& aFont) const
    {
    TInt addedWidth(0);
    if ( IsEditable() )
        {
        TInt rightHighlight(0);
        TAknTextDecorationMetrics metrics( &aFont );
        // Width is increased only by leftHighlight.  This is a compromise in that glyphs already have 
        // spacing within them to achieve character spacing. This spacing is generally (for numbers) on the right. 
        metrics.GetLeftAndRightMargins( addedWidth, rightHighlight ); // rightHighlight is not used
        }
    return addedWidth;
    }

const TDesC& CEikMfneField::FieldText() const
    {
    return Text();
    }

// CEikMfneSeparator

CEikMfneSeparator::CEikMfneSeparator(HBufC* aText)
    :iText(aText)
    {
    __DECLARE_NAME(_S("CEikMfneSeparator"));
    }

EXPORT_C CEikMfneSeparator::~CEikMfneSeparator()
    {
    delete iText;
    }

EXPORT_C CEikMfneSeparator* CEikMfneSeparator::NewL(TResourceReader& aResourceReader)
    {
    HBufC* text=aResourceReader.ReadHBufCL();
    CleanupStack::PushL(text);
    CEikMfneSeparator* separator=NewL(text);
    CleanupStack::Pop();
    return separator;
    }

EXPORT_C CEikMfneSeparator* CEikMfneSeparator::NewL(HBufC* aText)
    {
    return new(ELeave) CEikMfneSeparator(aText);
    }

EXPORT_C void CEikMfneSeparator::SetText(HBufC* aText)
    {
    __ASSERT_ALWAYS(iText==NULL, Panic(EEikPanicMfneSeparatorTextHasAlreadyBeenSet));
    iText=aText;
    }

TInt CEikMfneSeparator::MaximumWidthInPixels(const CFont& aFont, TBool)
    {
    return aFont.TextWidthInPixels(*iText)+AdditionalWidthForHighlights(aFont);
    }

TCoeInputCapabilities CEikMfneSeparator::InputCapabilities() const
    {
    return TCoeInputCapabilities(TCoeInputCapabilities::ENone);
    }

const TDesC& CEikMfneSeparator::Text() const
    {
    return *iText;
    }



// CEikMfneNumber

CEikMfneNumber::CEikMfneNumber(TInt aMinimumValue, TInt aMaximumValue, TUint32 aFlags)
    :iMinimumValue(aMinimumValue),
     iMaximumValue(aMaximumValue),
     iFlags(aFlags),
     iDigitType(EDigitTypeWestern)
    {
    TBuf<16> minText;
    minText.AppendNum(aMinimumValue);
    TBuf<16> maxText;
    maxText.AppendNum(aMaximumValue);
    iMaxDigits=Max(minText.Length(),maxText.Length());

    iMaxDigitsMinimumValue=minText.Length();
    iMaxDigitsMaximumValue=maxText.Length();

    __DECLARE_NAME(_S("CEikMfneNumber"));
    __ASSERT_ALWAYS(aMinimumValue<=aMaximumValue, Panic(EEikPanicMfneBadNumberMinimumAndMaximum1));
    __ASSERT_ALWAYS((aFlags&~EPublicallySettableFlags)==0, Panic(EEikPanicMfneBadNumberFlags));
    }

EXPORT_C CEikMfneNumber::~CEikMfneNumber()
    {
    delete iText;
    }

EXPORT_C CEikMfneNumber* CEikMfneNumber::NewL(const CFont& aFont, TResourceReader& aResourceReader)
    {
#pragma warning (disable : 4127)
    __ASSERT_DEBUG((EFillWithLeadingZeros==EEikMfneNumberFlagFillWithLeadingZeros) &&
                   (EPreserveOldWidthBeforeEditing==EEikMfneNumberFlagPreserveOldWidthBeforeEditing) &&
                   (EFillWithLeadingZeros==EEikMfneNumberFlagFillWithLeadingZeros), Panic(EEikPanicMfneInconsistentNumberFlags));
#pragma warning (default : 4127)
    TInt minimumValue=aResourceReader.ReadInt32();
    TInt maximumValue=aResourceReader.ReadInt32();
    TUint32 flags=aResourceReader.ReadUint8();
    return NewL(aFont, minimumValue, maximumValue, maximumValue, flags);
    }

EXPORT_C CEikMfneNumber* CEikMfneNumber::NewL(const CFont& aFont, TInt aMinimumValue, TInt aMaximumValue, TInt aInitialValue, TUint32 aFlags)
    {
    __ASSERT_ALWAYS((aInitialValue>=aMinimumValue) && (aInitialValue<=aMaximumValue), Panic(EEikPanicMfneBadNumberMinimumAndMaximum2));
    CEikMfneNumber* number=new(ELeave) CEikMfneNumber(aMinimumValue, aMaximumValue, aFlags);
    CleanupStack::PushL(number);
    number->iText=HBufC::NewL(number->MaximumNumberOfDigits()+((aMinimumValue<0)? 1: 0));
    number->SetValue(aInitialValue, aFont);
    number->ConstructL();
    CleanupStack::Pop();
    return number;
    }

void CEikMfneNumber::ConstructL()
    {
    CEikonEnv* eikEnv=CEikonEnv::Static();
    iNudgeCharMinus=eikEnv->NudgeCharMinus();
    iNudgeCharPlus=eikEnv->NudgeCharPlus();
    }

EXPORT_C void CEikMfneNumber::SetMinimumAndMaximum(TInt aMinimumValue, TInt aMaximumValue, const CFont& aFont)
    {
    TBuf<16> minText;
    minText.AppendNum(aMinimumValue);
    TBuf<16> maxText;
    maxText.AppendNum(aMaximumValue);
    TInt minTextLength=minText.Length();
    TInt maxTextLength=maxText.Length();
    __ASSERT_ALWAYS((aMinimumValue<=aMaximumValue) &&
                    (minTextLength<=iMaxDigits) &&
                    (maxTextLength<=iMaxDigits), Panic(EEikPanicMfneBadNumberMinimumAndMaximum3));
    //__ASSERT_ALWAYS((aMinimumValue<=aMaximumValue) &&
    //              (aMinimumValue>=iMinimumValueCateredForInFieldWidth) &&
    //              (aMaximumValue<=iMaximumValueCateredForInFieldWidth), Panic(EEikPanicMfneBadNumberMinimumAndMaximum3));
    iMinimumValue=aMinimumValue;
    iMaximumValue=aMaximumValue;
    TInt value=ValueFromText();
    if (value<aMinimumValue||!NumberOfDigits())
        SetTextToValue(aMinimumValue, aFont);
    else if (value>aMaximumValue)
        SetTextToValue(aMaximumValue, aFont);
    }

EXPORT_C void CEikMfneNumber::GetMinimumAndMaximum(TInt& aMinimumValue, TInt& aMaximumValue) const
    {
    aMinimumValue=iMinimumValue;
    aMaximumValue=iMaximumValue;
    }

EXPORT_C void CEikMfneNumber::SetValue(TInt aValue, const CFont& aFont)
    {
    // This is part of the trailing zeros and TInt fix. We have to do this to
    // allow setting values like "050" by passing 50 to this function in case
    // the trailing zeros are on.
    if ( iFlags & EFillWithTrailingZeros )
        {
        TInt val = aValue;
        TInt count = 0;
        for(;val;val/=10) count ++; 
        if (MaximumNumberOfDigits()-count < 0) count = MaximumNumberOfDigits();
        iText->Des().Fill( TText('1'), MaximumNumberOfDigits() );
        iText->Des().Fill( ZeroCharacter(), MaximumNumberOfDigits() - count);
        }
        
    SetTextToValue(aValue, aFont);
    }

EXPORT_C TInt CEikMfneNumber::Value() const
    {
    TInt value( -1 );
    if (NumberOfDigits() > 0)
        {
        value = ValueFromText();
        }    
    if (value < iMinimumValue)
        {
        value = iMinimumValue;
        }
    if (value > iMaximumValue)
        {       
        value = iMaximumValue;
        }
    return value;
    }

EXPORT_C TBool CEikMfneNumber::IsValid() const
    {
    return (NumberOfDigits() != 0);
    }

TChar CEikMfneNumber::NormalizeDigit(TChar aChar)
/**
NormalizeDigit
Normalises a character to the current digit type.
@internalComponent
@return TChar of  if this character is a numeral
@param aCharacter character to be tested 
*/
    {
    TBuf<1> buf;
    buf.Append(aChar);
    NumberConversion::ConvertDigits(buf, iDigitType);
    return buf[0];
    }

TText CEikMfneNumber::ZeroCharacter() const
        {
        return TText(iDigitType);
        }

TText CEikMfneNumber::NegativeCharacter() const
        {
        return TText('-');
        }

TBool CEikMfneNumber::IsTextNull() const
    {
    return ( Text().Length() < 1 );
    }

TInt CEikMfneNumber::MaximumWidthInPixels(const CFont& aFont, TBool aShrinkToMinimumSize)
    {
    if (aShrinkToMinimumSize)
        {
        //iMinimumValueCateredForInFieldWidth=iMinimumValue;
        //iMaximumValueCateredForInFieldWidth=iMaximumValue;
        TBuf<16> minText;
        minText.AppendNum(iMinimumValue);
        TBuf<16> maxText;
        maxText.AppendNum(iMaximumValue);

        iMaxDigits=Max(2,Max(minText.Length(),maxText.Length()));
        }
    return (MaximumNumberOfDigits()*TFindWidthOfWidestDigitType(DigitType()).MaximumWidthInPixels(aFont)) + AdditionalWidthForHighlights(aFont) /*+((iMinimumValueCateredForInFieldWidth<0)? aFont.TextWidthInPixels(_L("-")): 0)*/;
    }

TCoeInputCapabilities CEikMfneNumber::InputCapabilities() const
    {
    TUint inputCapabilities=TCoeInputCapabilities::ENone;
    if (iMinimumValue<0)
        {
        inputCapabilities|=TCoeInputCapabilities::EWesternNumericIntegerNegative;
        }
    if (iMaximumValue>0)
        {
        inputCapabilities|=TCoeInputCapabilities::EWesternNumericIntegerPositive;
        }
    return TCoeInputCapabilities(inputCapabilities);
    }

const TDesC& CEikMfneNumber::Text() const
    {
    if (IsUninitialised())
        return KNullDesC();
    else
        return *iText;
    }

TBool CEikMfneNumber::IsEditable() const
    {
    return ETrue;
    }

CEikMfneField::THighlightType CEikMfneNumber::HighlightType() const
    {
    return (iFlags&EIsBeingEditedWithCursor)? ECursor: EInverseVideo;
    }

void CEikMfneNumber::HandleKey(const CFont& aFont, const TKeyEvent& aKeyEvent, TBool aInterpretLeftAndRightAsEarEvents, TBool& aDataAltered, TInt& aHighlightIncrement)
    {
    TChar ch=aKeyEvent.iCode;

    TPtr text=iText->Des();
    TInt textLength=text.Length();
    TBool nudgeTen=EFalse;
    switch (ch)
        {
    case EKeyLeftArrow:
    case EKeyRightArrow:
        if (!aInterpretLeftAndRightAsEarEvents)
            HandleLeftOrRightArrow(ch, aDataAltered, aHighlightIncrement);
        else
            {
            const TInt nudge=(nudgeTen? 10 : 1);
            TInt newValue;


            if ((textLength==0) || ! EikMfneUtils::IsAnyDigit( TChar(text[textLength-1]) ) )
                newValue=0;
            else
                newValue=ValueFromText()+((ch==EKeyLeftArrow)? -nudge: nudge);
            if (newValue>=iMaximumValue)
                newValue=iMaximumValue;
            else if (newValue<=iMinimumValue)
                newValue=iMinimumValue;
            SetTextToValue(newValue, aFont);
            aHighlightIncrement=0;// Always highlight the field being nudged
            aDataAltered=ETrue;
            SetUninitialised(EFalse);
            iFlags&=~EIsBeingEditedWithCursor;
            }
        break;
    case EKeyDownArrow:
    case EKeyUpArrow:
        {
        if (IsUninitialised())
            {
            SetUninitialised(EFalse);
            }
        else
            {
                const TInt nudge=(nudgeTen? 10 : 1);
                TInt newValue;


                if ((textLength==0) || ! EikMfneUtils::IsAnyDigit( TChar(text[textLength-1])) )
                    newValue=0;
                else
                        newValue=ValueFromText()+((ch==EKeyDownArrow)? -nudge: nudge);
       
            if (newValue>iMaximumValue)
                        newValue=iMinimumValue;
                else if (newValue<iMinimumValue)
                    newValue=iMaximumValue;
                SetTextToValue(newValue, aFont);
                aHighlightIncrement=0;// Always highlight the field being nudged
                aDataAltered=ETrue;
                iFlags&=~EIsBeingEditedWithCursor;
            }
        aDataAltered=ETrue;
        }
        break;
    case EKeyBackspace:
        if (textLength)
            {
            if (iFlags&EIsBeingEditedWithCursor)
                {
                text.SetLength(textLength-1);
                }
            else
                {
                iFlags|=EIsBeingEditedWithCursor;
                text.SetLength(0);
                }
            if (text==KMinus)
                text.SetLength(0);
            aDataAltered=ETrue;
            SetUninitialised(EFalse);
            }
        break;
    case '+':
        break;
    case '-':
        if (iMinimumValue < 0)
            {
            if (~iFlags&EIsBeingEditedWithCursor)
                {
                iFlags|=EIsBeingEditedWithCursor;
                text.SetLength(0);
                }
            if (text.Length()==0)
                {
                text.Append('-');
                aDataAltered=ETrue;
                SetUninitialised(EFalse);
                }
            }
        break;
    default:
        if ( EikMfneUtils::IsAnyDigit(ch) )
            {
            if (~iFlags&EIsBeingEditedWithCursor)
                {
                iFlags|=EIsBeingEditedWithCursor;
                text.SetLength(0);
                }
            __ASSERT_DEBUG(NumberOfDigits()<MaximumNumberOfDigits(),Panic(EEikPanicMfneNumberTextInBadState));
            text.Append(NormalizeDigit(ch) );
            aDataAltered=ETrue;
            SetUninitialised(EFalse);
            
            // Space reserved for minus sign is not supposed to be shown, so if
            // the value is >= 0 we must subtract the space for minus in case
            // the minimum possible value is negative..
            if ( NumberOfDigits() >= 
                ( ValueFromText() >= 0 && iMinimumValue < 0 ? MaximumNumberOfDigits() - 1 : MaximumNumberOfDigits() ) )
                {
                aHighlightIncrement=1;
                }
            }
        break;
        }
    }

void CEikMfneNumber::HandleDeHighlight(const CFont& aFont, CEikonEnv& aEikonEnv, TBool& aDataAltered, TBool& aError)
    {
    iFlags&=~EIsBeingEditedWithCursor;
    if (NumberOfDigits()==0)
        {
        SetTextToValue(iMinimumValue, aFont);
        aDataAltered=ETrue;
        aError=ETrue;
        aEikonEnv.InfoMsg(R_EIK_TBUF_NO_NUMBER_ENTERED);
        return;
        }
    TInt value=ValueFromText();
    if ((value<iMinimumValue) || (value>iMaximumValue))
        {
        TBool succeeded = ConvertsIntoValidValue(value);
        if ( !succeeded )
            {
            aDataAltered=ETrue;
            aError=ETrue;
            if (value<iMinimumValue)
                {
                SetTextToValue(iMinimumValue, aFont);
                CEikMfne::InvalidFieldAlert();
                aEikonEnv.InfoMsg(R_EIK_TBUF_NUMBER_BELOW_MIN, iMinimumValue);
                return;
                }
            if (value>iMaximumValue) 
                {
                CEikMfne::InvalidFieldAlert();
                SetTextToValue(iMaximumValue, aFont);
                aEikonEnv.InfoMsg(R_EIK_TBUF_NUMBER_ABOVE_MAX, iMaximumValue);
                return;
                }
            }
        }
    TBuf<128> oldText=*iText;
    TBool unInit = IsUninitialised();
    SetTextToValue(value, aFont);
    if (oldText!=*iText)
        aDataAltered=ETrue;
    else
        SetUninitialised(unInit);
    }

TInt CEikMfneNumber::MaximumNumberOfDigits() const
    {
    return Max(2,iMaxDigits);
    }

TInt CEikMfneNumber::NumberOfDigits() const
    {
    return iText->Length();
    }

void CEikMfneNumber::SetTrailingZeros()
    {
    iFlags |= EFillWithTrailingZeros;
    }

void CEikMfneNumber::SetTextToValue(TInt aValue, const CFont& /*aFont*/)
    {
    __ASSERT_DEBUG((aValue>=iMinimumValue) && (aValue<=iMaximumValue), Panic(EEikPanicMfneNumberOutsidePermittedRange));
    
    SetUninitialised(EFalse);

    TPtr textPtr       = iText->Des();
    TInt maxNumbers    = MaximumNumberOfDigits();
    TInt leadingZeros  = 0;
    TText zeroChar     = ZeroCharacter();
    TText negChar      = NegativeCharacter();

    // The problem here is that we have no way of knowing through TInt aValue
    // if the number that is wanted is 005, 05, or 5. That would be fine
    // as long as trailing zeros are not used, but with the trailing
    // zeros one has to be able to enter "050" by typing just "05".
    // When this is typed and the field changes, this function gets called with
    // aValue of 5, and we don't know if that should really be "500", "050" or
    // "005", because all those values could be possible in a number with
    // trailing zeros. The following tries to overcome this limitation..
    TLex lex( textPtr );
    TChar chr;

    while ( chr = lex.Get() )
        {
        if ( chr == zeroChar )
            {
            leadingZeros++;
            }
        else if ( chr != negChar )
            {
            break; // We break at first char that's not '-' or zero
            }
        }
        
    textPtr.SetLength( 0 );

    if ( iMinimumValue < 0 )
        {
        maxNumbers--;

        // Has to be done here, because we have to pad between
        // the negative char and the number.
        if ( aValue < 0 )
            {
            textPtr.Append( negChar );
            aValue = -aValue;
            }        
        }
        
    // From this on we deal with western digits, and convert at the end
    zeroChar = TText( '0' );
                
    if ( iFlags & EFillWithTrailingZeros ) // trailing takes precedence
        {
        HBufC* formatStringBuf = HBufC::New( leadingZeros + 10 );
        
        if ( !formatStringBuf )
            {
            textPtr.Num( aValue );
            }
        else
            {
            TPtr formatString = formatStringBuf->Des();
            
            if ( aValue != 0 && leadingZeros )
                {
                formatString.Fill( zeroChar, leadingZeros );
                maxNumbers -= leadingZeros;
                }

            _LIT( KFormatString, "%-**d" );
            formatString.Append( KFormatString );
            textPtr.AppendFormat( formatString, zeroChar, maxNumbers, aValue );
            delete formatStringBuf;
            }
        }
    else if ( iFlags & EFillWithLeadingZeros )
        {
        _LIT( KFormatString, "%+**d" );
        textPtr.Format( KFormatString, zeroChar, maxNumbers, aValue );
        }
    else
        {
        textPtr.Num( aValue );
        }
    
    NumberConversion::ConvertDigits( textPtr, iDigitType );
    
    if ( iFlags & EPreserveOldWidthBeforeEditing )
        {
        iMinimumWidthInPixels = 0; //WidthInPixels(aFont);    
        }
    }

TInt CEikMfneNumber::ValueFromText() const
    {
//  __ASSERT_DEBUG(NumberOfDigits(), Panic(EEikPanicMfneNumberHasNoDigits));
    if (!NumberOfDigits())
        return iMinimumValue;
    TInt i=0;
    TInt valueFromText=0;
    TBool isNegative=EFalse;
    switch ((*iText)[i])
        {
    case '-':
        ++i;
        isNegative=ETrue;
        break;
        }
    TInt textLength=iText->Length();
    for (; i<textLength; ++i)
        {
        TText digit=(*iText)[i];
        __ASSERT_DEBUG(EikMfneUtils::IsAnyDigit( TChar(digit)), Panic(EEikPanicMfneDigitExpected));
        if (i>=textLength-2)
            {
            if (!isNegative && valueFromText>(KMaxTInt32/10))
                return KMaxTInt32;
            else if (isNegative && (-valueFromText)<(KMinTInt32/10))
                return KMinTInt32;
            }
        valueFromText=(valueFromText*10)+(TInt)(digit-ZeroCharacter());
        }
    if (isNegative==EFalse && valueFromText<0)
        return KMaxTInt32;// Deals with overflow
    if (isNegative)
        valueFromText=-valueFromText;
    return valueFromText;
    }

TBool CEikMfneNumber::ConvertsIntoValidValue(TInt& aValue) const
    {
    if ((iFlags&ERepresentsYear) && (aValue>=0) && (aValue<100))
        {
        TTime homeTime;
        homeTime.HomeTime();
        TInt currentYear=homeTime.DateTime().Year();
        if (currentYear>0)
            {
            TInt yearsSinceStartOfCurrentCentury=currentYear%100;
            TInt newValue=(currentYear-yearsSinceStartOfCurrentCentury)+aValue;
            // If two digit year is given, 2000 + aValue is expected.
            aValue=newValue;
            if ((newValue>=iMinimumValue) && (newValue<=iMaximumValue))
                {
                return ETrue;
                }
            }
        }
    return EFalse;
    }

EXPORT_C void CEikMfneNumber::SetUninitialised(TBool aUninitialised)
    {
    if (aUninitialised)
        iFlags |= EIsUninitialised;
    else
        iFlags &= ~EIsUninitialised;
    }

EXPORT_C TBool CEikMfneNumber::IsUninitialised() const
    {
    return iFlags & EIsUninitialised;
    }


EXPORT_C void CEikMfneNumber::RefreshDigitType( const CFont& aFont)
/**
Derive digit display type from locale information. This call sets the digit type based on current device settings.
It can be called after construction to make the field display correct numeral (if it is required.
Or can be called any number of times in response to Tlocale change. For dynamic update.
@publishedComponent
@since 2.0
@lib eikctl.lib
@param a locale structure which will be used to decide on the digit display.
@param aFont the font of the MFNE which owns this class
@post the next time the field is drawn it will display the number using digit type derived from the locale information.
*/
    {
    SetDigitType( AknTextUtils::NumericEditorDigitType(), aFont);
    }

EXPORT_C void CEikMfneNumber::SetDigitType(TDigitType aDigitType, const CFont& aFont)
/**
Set Numeral Type for displaying Numbers type for International digit support
@publishedComponent
@since 2.0
@lib eikctl.lib
@param aDigitType Set the numeral type for this field to display numbers in.
@param aFont the font of the MFNE which owns this class
@post the next time the field is drawn it will display the number using the supplied digit type.
*/
    {
    TInt value = Value();
    iDigitType = aDigitType;
    SetTextToValue(value, aFont);
    }

EXPORT_C TDigitType CEikMfneNumber::DigitType() const
/**
retrieve the current numeral type of the field
@publishedComponent
@since 2.0
@lib eikctl.lib
@return the current numeral display type.
*/
    {
    return iDigitType;
    }

// CEikMfneSymbol

CEikMfneSymbol::CEikMfneSymbol(TInt aNumSymbolicItems)
    :iNumSymbolicItems(aNumSymbolicItems),
     iCurrentSymbolicItem(0)
    {
    __DECLARE_NAME(_S("CEikMfneSymbol"));
    __ASSERT_ALWAYS(aNumSymbolicItems>1, Panic(EEikPanicMfneTooFewSymbolicItems));
    }

EXPORT_C CEikMfneSymbol::~CEikMfneSymbol()
    {
    if (iSymbolicItems)
        {
        for (TInt i=0; i<iNumSymbolicItems; ++i)
            delete iSymbolicItems[i];
        delete [] iSymbolicItems;
        }
    }

EXPORT_C CEikMfneSymbol* CEikMfneSymbol::NewL(TResourceReader& aResourceReader)
    {
    TInt numSymbols=aResourceReader.ReadUint8();
    CEikMfneSymbol* symbol=NewL(numSymbols);
    CleanupStack::PushL(symbol);
    for (TInt i=0; i<numSymbols; ++i)
        symbol->AddSymbolicItem(CItem::NewL(aResourceReader), i==0);
    CleanupStack::Pop();
    return symbol;
    }

EXPORT_C CEikMfneSymbol* CEikMfneSymbol::NewL(TInt aNumSymbolicItems)
    {
    CEikMfneSymbol* symbol=new(ELeave) CEikMfneSymbol(aNumSymbolicItems);
    CleanupStack::PushL(symbol);
    symbol->iSymbolicItems=new(ELeave) CItem*[aNumSymbolicItems];
    for (TInt i=0; i<aNumSymbolicItems; ++i)
        symbol->iSymbolicItems[i]=NULL;
    CleanupStack::Pop();
    return symbol;
    }

EXPORT_C void CEikMfneSymbol::AddSymbolicItem(CItem* aSymbolicItem, TBool aMakeCurrent)
    {
    __ASSERT_DEBUG(iSymbolicItems, Panic(EEikPanicMfneSymbolicItemArrayNotCreated));
    for (TInt i=0; i<iNumSymbolicItems; ++i)
        if (iSymbolicItems[i]==NULL)
            {
            iSymbolicItems[i]=aSymbolicItem;
            if (aMakeCurrent)
                SetCurrentSymbolicItem(i);
            return;
            }
#if defined(_DEBUG)
    Panic(EEikPanicMfneTooManySymbolicItemsAdded);
#endif
    }

EXPORT_C void CEikMfneSymbol::SetCurrentSymbolicItemToId(TInt aId)
    {
    for (TInt i=0; i<iNumSymbolicItems; ++i)
        if (iSymbolicItems[i]->iId==aId)
            {
            SetCurrentSymbolicItem(i);
            SetUninitialised(EFalse);
            return;
            }
#if defined(_DEBUG)
    Panic(EEikPanicMfneIdOfSymbolicItemNotFound);
#endif
    }

EXPORT_C TInt CEikMfneSymbol::IdOfCurrentSymbolicItem() const
    {
    return iSymbolicItems[CurrentSymbolicItem()]->iId;
    }

TInt CEikMfneSymbol::MaximumWidthInPixels(const CFont& aFont, TBool)
    {
    TInt maximumWidthInPixels=0;
    for (TInt i=0; i<iNumSymbolicItems; ++i)
        {
        TInt widthInPixels=aFont.TextWidthInPixels(*iSymbolicItems[i]->iText);
        if (maximumWidthInPixels<widthInPixels)
            maximumWidthInPixels=widthInPixels;
        }

    /* In the uninitialised state iMinimumWidthInPixels is set to the max with so that when focussed in the uninitialised state the
     * whole space is highlighted.  Note that when the state becomes not uninitialised iMinimumwidth... is set back to zero.
     */
    if (IsUninitialised())
        iMinimumWidthInPixels = maximumWidthInPixels + AdditionalWidthForHighlights(aFont);
    else
        iMinimumWidthInPixels = 0;
    return maximumWidthInPixels;
    }

TCoeInputCapabilities CEikMfneSymbol::InputCapabilities() const
    {
    TUint inputCapabilities=TCoeInputCapabilities::EWesternAlphabetic;
    for (TInt i=0; i<iNumSymbolicItems; ++i)
        {
        const TCharF keyToMatch=iSymbolicItems[i]->iKeyToMatch;
        if ((keyToMatch<TCharF('A')) || (keyToMatch>TCharF('Z'))) // a crude test, but it's probably sufficient
            {
            inputCapabilities=TCoeInputCapabilities::EAllText;
            }
        }
    return TCoeInputCapabilities(inputCapabilities);
    }

const TDesC& CEikMfneSymbol::Text() const
    {
    if (IsUninitialised())
        return KNullDesC();
    else
        return *iSymbolicItems[CurrentSymbolicItem()]->iText;
    }

TBool CEikMfneSymbol::IsEditable() const
    {
    return ETrue;
    }

CEikMfneField::THighlightType CEikMfneSymbol::HighlightType() const
    {
    return EInverseVideo;
    }

void CEikMfneSymbol::HandleKey(const CFont&, const TKeyEvent& aKeyEvent, TBool, TBool& aDataAltered, TInt& aHighlightIncrement)
    {
    TChar ch=aKeyEvent.iCode;
    switch (ch)
        {
        case EKeyLeftArrow:
        case EKeyRightArrow:
            HandleLeftOrRightArrow(ch, aDataAltered, aHighlightIncrement);
            break;
        case EKeyPowerOff:
        case EKeyPhoneEnd:
        case EKeyApplication:
            break;
        default:
            if (IsUninitialised())
                SetUninitialised(EFalse);
            else
                SetCurrentSymbolicItem((CurrentSymbolicItem()+1)%iNumSymbolicItems);
            aDataAltered=ETrue;
            break;
        }
    }

void CEikMfneSymbol::HandleDeHighlight(const CFont&, CEikonEnv&, TBool&, TBool&)
    {
    }



/*
 * The unilitialised flag uses the topmost bit of iCurrentSymbolicItem.  This is to preserve the size of the class for BC.
 * It's important that iCurrentSymbolic item is never negative though as it is an index that should not happen.  
 *
 */
const TInt KMfneSymbolUninitialisedBit = 0xf0000000;
EXPORT_C void CEikMfneSymbol::SetUninitialised(TBool aUninitialised)
    {
    if (aUninitialised)
        {
        iCurrentSymbolicItem |= KMfneSymbolUninitialisedBit;
        }
    else
        {
        iCurrentSymbolicItem &= ~KMfneSymbolUninitialisedBit;
        iMinimumWidthInPixels = 0;
        }
    }

EXPORT_C TBool CEikMfneSymbol::IsUninitialised() const
    {
    return iCurrentSymbolicItem & KMfneSymbolUninitialisedBit;
    }

TInt CEikMfneSymbol::CurrentSymbolicItem() const
    {
    return iCurrentSymbolicItem & ~KMfneSymbolUninitialisedBit;
    }

void CEikMfneSymbol::SetCurrentSymbolicItem(TInt aCurrentSymbolicItem)
    {
    __ASSERT_ALWAYS( aCurrentSymbolicItem >=0, Panic( EEikPanicMfneIdOfSymbolicItemNotFound ) ) ;
    TInt uninitialised = iCurrentSymbolicItem & KMfneSymbolUninitialisedBit;
    iCurrentSymbolicItem = aCurrentSymbolicItem | uninitialised;
    }

// CEikMfneSymbol::CItem

CEikMfneSymbol::CItem::CItem(TInt aId, TChar aKeyToMatch, HBufC* aText)
    :iId(aId),
     iKeyToMatch(aKeyToMatch),
     iText(aText)
    {
    __DECLARE_NAME(_S("CItem"));
    }

EXPORT_C CEikMfneSymbol::CItem::~CItem()
    {
    delete iText;
    }

EXPORT_C CEikMfneSymbol::CItem* CEikMfneSymbol::CItem::NewL(TResourceReader& aResourceReader)
    {
    TInt id=aResourceReader.ReadInt32();
    TChar keyToMatch=aResourceReader.ReadUint16();
    HBufC* text=aResourceReader.ReadHBufCL();
    CleanupStack::PushL(text);
    CItem* symbolicItem=NewL(id, keyToMatch, text);
    CleanupStack::Pop();
    return symbolicItem;
    }

EXPORT_C CEikMfneSymbol::CItem* CEikMfneSymbol::CItem::NewL(TInt aId, TChar aKeyToMatch, HBufC* aText)
    {
    return new(ELeave) CItem(aId, aKeyToMatch, aText);
    }

EXPORT_C void CEikMfneSymbol::CItem::SetText(HBufC* aText)
    {
    __ASSERT_DEBUG(iText==NULL, Panic(EEikPanicMfneSymbolicItemTextHasAlreadyBeenSet));
    iText=aText;
    }

//
// CEikMfne
//

// Enumerations use to implement the flag setter/getter methods used with CEikMfne::iFlags
enum TEikMfneFlagIndex
    {
    EConsumeUpAndDownKeysIndex = 0,
    ESkinningBackground,
    ESkinBackGroundControlContextSetByApi,
    ESuppressBackgroundDrawing,
    EUseOverrideColors
    };

EXPORT_C CEikMfne::CEikMfne()
    :CEikBorderedControl(TGulBorder(TGulBorder::ESingleGray)),
     iCurrentField(ENullIndex)
    {
    __DECLARE_NAME(_S("CEikMfne"));
    AKNTASHOOK_ADD( this, "CEikMfne" );
    }

EXPORT_C CEikMfne::~CEikMfne()
    {
    AKNTASHOOK_REMOVE();
    if (iFields)
        {
        if (IsFocused() && (iCurrentField!=ENullIndex) && (iFields[iCurrentField]->HighlightType()==CEikMfneField::ECursor))
            HideCursor();
        for (TInt i=0; i<iNumFields; ++i)
            delete iFields[i];
        delete [] iFields;
        }
    delete iExtension;
    }

EXPORT_C void CEikMfne::CreateFieldArrayL(TInt aNumFields)
    {
    __ASSERT_DEBUG(iFields==NULL, Panic(EEikPanicMfneFieldArrayAlreadyCreated));
    __ASSERT_DEBUG(aNumFields>0, Panic(EEikPanicMfneTooFewFields));
    iNumFields=aNumFields;
    iFields=new(ELeave) CEikMfneField*[aNumFields];
    for (TInt i=0; i<aNumFields; ++i)
        iFields[i]=NULL;
    // Set the flag that says background skin drawing will be done
    EvaluateSkinningBackground();
    CreateExtensionIfRequiredL();
    }
EXPORT_C void CEikMfne::ResetFieldArray()
    {
    if (iFields)
        {
        for (TInt i=0; i<iNumFields; ++i)
            delete iFields[i];
        delete [] iFields;
        iFields = NULL;
        iNumFields = 0;
        iCurrentField = ENullIndex;
        }   
    }

EXPORT_C void CEikMfne::AddField(CEikMfneField* aField)
    {
    __ASSERT_DEBUG(iFields, Panic(EEikPanicMfneFieldArrayNotCreated));
    for (TInt i=0; i<iNumFields; ++i)
        if (iFields[i]==NULL)
            {
            iFields[i]=aField;
            if ((iCurrentField==ENullIndex) && aField->IsEditable())
                iCurrentField=i;
            return;
            }
#if defined(_DEBUG)
    Panic(EEikPanicMfneTooManyFieldsAdded);
#endif
    }

EXPORT_C TMargins CEikMfne::BorderMargins() const
    {
    return iBorder.Margins();
    }

EXPORT_C TTime CEikMfne::ReadTime(TResourceReader& aResourceReader)
    {
    TInt second=aResourceReader.ReadUint8();
    TInt minute=aResourceReader.ReadUint8();
    TInt hour=aResourceReader.ReadUint8();
    return TTime(TDateTime(0, EJanuary, 0, hour, minute, second, 0));
    }

EXPORT_C TTime CEikMfne::ReadDate(TResourceReader& aResourceReader)
    {
    TInt day=aResourceReader.ReadUint8();
    TMonth month=(TMonth)aResourceReader.ReadUint8();
    TInt year=aResourceReader.ReadInt16();
    return TTime(TDateTime(year, month, day, 0, 0, 0, 0));
    }

EXPORT_C TTime CEikMfne::ReadTimeAndDate(TResourceReader& aResourceReader)
    {
    TInt second=aResourceReader.ReadUint8();
    TInt minute=aResourceReader.ReadUint8();
    TInt hour=aResourceReader.ReadUint8();
    TInt day=aResourceReader.ReadUint8();
    TMonth month=(TMonth)aResourceReader.ReadUint8();
    TInt year=aResourceReader.ReadInt16();
    return TTime(TDateTime(year, month, day, hour, minute, second, 0));
    }

EXPORT_C TTimeIntervalSeconds CEikMfne::ReadDuration(TResourceReader& aResourceReader)
    {
    TInt seconds=aResourceReader.ReadInt32();
    __ASSERT_ALWAYS(seconds>=0, Panic(EEikPanicMfneNegativeDuration));
    return TTimeIntervalSeconds(seconds);
    }

EXPORT_C TTimeIntervalSeconds CEikMfne::ReadTimeOffset(TResourceReader& aResourceReader)
    {
    TInt seconds=aResourceReader.ReadInt32();
    return TTimeIntervalSeconds(seconds);
    }

void CEikMfne::InvalidFieldAlert()
/**
Display alert to user, when input is not a valid number (too big or too small etc).
Current implementation is to play a sound alert.
@publishedComponent
@since 2.0
*/
    {
    CAknKeySoundSystem* soundPlayer = (static_cast<CAknAppUi*>(CEikonEnv::Static()->AppUi()))->KeySounds();
    if(soundPlayer)
        {
        soundPlayer->PlaySound(EAvkonSIDWarningTone); 
        }
    }

void CEikMfne::LeaveWithAlert(TInt /*aResourceId*/)
/**
This is to aid the old code in the derived classes which called LeaveWithInfoMsg().
Now they leave and play warning sound to indicate a problem/correction made by the editor.
@publishedComponent
@since 2.0
*/
    {
    InvalidFieldAlert();
    CBaActiveScheduler::LeaveNoAlert();
    }

EXPORT_C TTimeIntervalSeconds CEikMfne::Convert(const TTime& aTime)
    {
    TDateTime dateTime=aTime.DateTime();
    return TTimeIntervalSeconds(dateTime.Second()+(dateTime.Minute()*60)+(dateTime.Hour()*3600)+(dateTime.Day()*3600*24));
    }

EXPORT_C TTime CEikMfne::Convert(const TTimeIntervalSeconds& aTimeIntervalSeconds)
    {
    TInt timeIntervalSeconds=aTimeIntervalSeconds.Int();
    TInt second=timeIntervalSeconds%60;
    timeIntervalSeconds/=60;
    TInt minute=timeIntervalSeconds%60;
    timeIntervalSeconds/=60;
    TInt hour=timeIntervalSeconds%24;
    return TTime(TDateTime(0, EJanuary, 0, hour, minute, second, 0));
    }

EXPORT_C TKeyResponse CEikMfne::OfferKeyEventL(const TKeyEvent& aKeyEvent, TEventCode aType)
    {
    const TInt code=aKeyEvent.iCode;

    if ( !ConsumesUpAndDownKeys() && (code==EKeyDownArrow || code==EKeyUpArrow))
        return EKeyWasNotConsumed;

    if ((aType==EEventKey) && (iCurrentField!=ENullIndex) && iExtension &&
        ( aKeyEvent.iRepeats == 0 || code == EKeyLeftArrow || 
            code == EKeyRightArrow || code == EKeyDownArrow || code == EKeyUpArrow ) )
        {
        iExtension->iHighlightAll = EFalse;
        const CFont& font=*Font();
        CEikMfneField::THighlightType oldHighlightTypeOfOldCurrentField=iFields[iCurrentField]->HighlightType();
        TInt oldWidthInPixelsOfOldCurrentField=iFields[iCurrentField]->WidthInPixels(font);
        TBool dataAltered=EFalse;
        TInt highlightIncrement=0;
        iFields[iCurrentField]->HandleKey(font, aKeyEvent, iNumFields==1, dataAltered, highlightIncrement);
        TInt newCurrentField;
        for (TInt i=iCurrentField+highlightIncrement; ; i+=highlightIncrement)
            {
            if (i<0)
                i=iNumFields-1;
            else if (i>=iNumFields)
                i=0;
            if (iFields[i]->IsEditable())
                {
                newCurrentField=i;
                break;
                }
            }
        TBool error=EFalse;
        HandleInteraction(highlightIncrement, newCurrentField, oldWidthInPixelsOfOldCurrentField, oldHighlightTypeOfOldCurrentField, dataAltered, error);
        ReportUpdate();
        }
    return EKeyWasConsumed;
    }

EXPORT_C void CEikMfne::PrepareForFocusLossL()
    {
    if (iCurrentField!=ENullIndex)
        {
        TBool dataAltered=EFalse;
        TBool error=EFalse;
        HandleInteraction(ETrue, iCurrentField, iFields[iCurrentField]->WidthInPixels(*Font()), iFields[iCurrentField]->HighlightType(), dataAltered, error);
        ReportUpdate();
        }
    }

EXPORT_C TSize CEikMfne::MinimumSize()
    {
    return MfneSize(ETrue);
    }

EXPORT_C TCoeInputCapabilities CEikMfne::InputCapabilities() const
    {
#ifdef RD_SCALABLE_UI_V2
    TCoeInputCapabilities inputCapabilities(
        TCoeInputCapabilities::ENavigation, iExtension, NULL );
#else       
    TCoeInputCapabilities inputCapabilities(
        TCoeInputCapabilities::ENavigation );
#endif // RD_SCALABLE_UI_V2     
    
    for (TInt i=0; i<iNumFields; ++i)
        {
        inputCapabilities.MergeWith(iFields[i]->InputCapabilities());
        }

    //inputCapabilities.SetObjectProvider(const_cast<CEikMfne*>(this));
    if ( iExtension )
        {
        inputCapabilities.SetObjectProvider(
            iExtension->iExtendedInputCapabilitiesProvider );
        }
        
    return inputCapabilities;
    }

EXPORT_C void CEikMfne::FocusChanged(TDrawNow aDrawNow)
    {
    if (iCurrentField==ENullIndex)
        return;
    if ( IsFocused() && iExtension && !iExtension->iDisablePenInput && 
        iExtension->iLaunchPenInputAutomatic )
        {
        TRAP_IGNORE( LaunchPenInputL() );
        }
    if (iFields[iCurrentField]->HighlightType()==CEikMfneField::ECursor)
        {
        if (IsFocused())
            DrawCursor();
        else
            HideCursor();
        return;
        }
        
    if ( aDrawNow  && !(iExtension && iExtension->iDisablePenInput) )
        {
        ActivateGc();
        if ( CAknEnv::Static()->TransparencyEnabled() )
            {
            TRect ctrlRect = Rect();
        
            if ( !IsBackedUp() )
                {
                Window().Invalidate( ctrlRect );
                Window().BeginRedraw( ctrlRect );
                }
            // Just draw all, sacrifice perf a bit for vast simplicity
            DrawRange( PreparedGc(), -1, iNumFields );
            SystemGc().DiscardFont();
            Window().EndRedraw();
            
            }
        else
            {
            if (iAlignment == ELayoutAlignRight || iAlignment == ELayoutAlignCenter)
                { // need to redraw whole editor because right align
                DrawRange( PreparedGc(), -1, iNumFields );
                }
            else
                { // updating current field is enough
                DrawRange(PreparedGc(), iCurrentField,iCurrentField);    
                }
            SystemGc().DiscardFont();
            }
  
        DeactivateGc();
        }
    }

EXPORT_C void CEikMfne::HandleInteraction(TBool aHandleDeHighlight, TInt aNewCurrentField, TInt aOldWidthInPixelsOfOldCurrentField,
                                CEikMfneField::THighlightType aOldHighlightTypeOfOldCurrentField, TBool& aDataAltered, TBool& aError)
    {
    const CFont& font=*Font();
    TBool drawAllFields = ETrue;
    if ( aHandleDeHighlight && iExtension )
        {
        iFields[iCurrentField]->HandleDeHighlight(font, *iEikonEnv, aDataAltered, aError);
        iExtension->iValidateValueCallBack.CallBack();
        if (aError)
            aNewCurrentField=iCurrentField;
        else
            FieldIsAboutToBeDeHighlighted(iFields[iCurrentField], drawAllFields);
        }
    TInt oldCurrentField=iCurrentField;
    iCurrentField=aNewCurrentField;

    if ( aDataAltered )
        {
        ReportEventL(MCoeControlObserver::EEventStateChanged);  
        }
    
    CEikMfneField::THighlightType newHighlightTypeOfOldCurrentField=iFields[oldCurrentField]->HighlightType();
    CEikMfneField::THighlightType highlightTypeOfCurrentField=iFields[iCurrentField]->HighlightType();

    ActivateGc();
    if ( CAknEnv::Static()->TransparencyEnabled() )
        {
        if ( !IsBackedUp() )
            {
            TRect ctrlRect = Rect();
            Window().Invalidate( ctrlRect );
            Window().BeginRedraw( ctrlRect );
            }
        
        // Just draw all, sacrifice perf a bit for vast simplicity
        drawAllFields = ETrue;
        }

    SystemGc().Reset();
    CWindowGc& gc=PreparedGc();

    if (drawAllFields ||iAlignment == ELayoutAlignRight || iAlignment == ELayoutAlignCenter) // draw all fields in case of right align
        DrawRange(gc, -1, iNumFields);
    else
        {
        TInt newWidthInPixelsOfOldCurrentField=iFields[oldCurrentField]->WidthInPixels(font);
        TInt firstFieldToDraw=(aDataAltered ||
                               ((iCurrentField!=oldCurrentField) && (aOldHighlightTypeOfOldCurrentField==CEikMfneField::EInverseVideo)) ||
                               ((iCurrentField==oldCurrentField) && (aOldHighlightTypeOfOldCurrentField!=newHighlightTypeOfOldCurrentField)))?
                                                                    oldCurrentField: oldCurrentField+1;
        TInt lastFieldToDraw=(newWidthInPixelsOfOldCurrentField!=aOldWidthInPixelsOfOldCurrentField)? iNumFields: oldCurrentField;
        DrawRange(gc, firstFieldToDraw, lastFieldToDraw);
        if ((iCurrentField!=oldCurrentField) && ((iCurrentField<firstFieldToDraw) || (iCurrentField>lastFieldToDraw)))
            {
            __ASSERT_DEBUG(highlightTypeOfCurrentField==CEikMfneField::EInverseVideo, Panic(EEikPanicMfneInverseVideoHighlightTypeExpected));
            DrawRange(gc, iCurrentField, iCurrentField);
            }
        }
    if ( (aOldHighlightTypeOfOldCurrentField==CEikMfneField::ECursor) && 
        ((iCurrentField!=oldCurrentField) || 
        (newHighlightTypeOfOldCurrentField!=CEikMfneField::ECursor)))
        {
        __ASSERT_DEBUG(highlightTypeOfCurrentField!=CEikMfneField::ECursor, Panic(EEikPanicMfneBadCursorState1));
        HideCursor();
        }
    if ( (highlightTypeOfCurrentField==CEikMfneField::ECursor) && 
        ((aOldHighlightTypeOfOldCurrentField!=CEikMfneField::ECursor) || aDataAltered))
        {
        __ASSERT_DEBUG(iCurrentField==oldCurrentField, Panic(EEikPanicMfneBadCursorState2));
        DrawCursor();
        }
    gc.DiscardFont();
    if ( CAknEnv::Static()->TransparencyEnabled() && !IsBackedUp() )
        {
        Window().EndRedraw();
        }
    DeactivateGc();
    }

EXPORT_C void CEikMfne::FieldIsAboutToBeDeHighlighted(CEikMfneField*, TBool&)
    {
    }

EXPORT_C void CEikMfne::DrawNowAndLeaveWithTimeDateFormatInfoMsgL(TInt /*aResourceId*/, const TTime& /*aTimeDate*/) const
    {
    CEikMfne::InvalidFieldAlert();
    DrawNow();
//  AVKON does not show info messages, removed because they take up lots of time and space
    CBaActiveScheduler::LeaveNoAlert();
    }

EXPORT_C CEikMfneField* CEikMfne::Field(TInt aField) const
    {
    return (aField<0 || aField>=iNumFields)?NULL:iFields[aField];
    }

EXPORT_C void CEikMfne::SetMfneAlignment(TInt aAlignment)
    {
    if( !iExtension )
        {
        TRAPD( err, CreateExtensionIfRequiredL() );
        if ( err != KErrNone )
            {
            return;
            }
        }
    if ( aAlignment >= ELayoutAlignNone )
        {
        iAlignment = aAlignment;
        TUint capabilities = iExtension->
            iExtendedInputCapabilities->Capabilities();
        capabilities &= 
            ~( CAknExtendedInputCapabilities::KAknEditorAlignMask );
        switch( iAlignment )
            {
            case ELayoutAlignCenter:
                capabilities |= 
                    CAknExtendedInputCapabilities::EInputEditorAlignCenter;
                break;
            case ELayoutAlignLeft:
                capabilities |= 
                    CAknExtendedInputCapabilities::EInputEditorAlignLeft;
                break;
            case ELayoutAlignRight:
                capabilities |=
                    CAknExtendedInputCapabilities::EInputEditorAlignRight;
                break;
            case ELayoutAlignBidi:
                capabilities |=
                    CAknExtendedInputCapabilities::EInputEditorAlignBidi;
                break;
            default:
                break;
            }
        iExtension->iExtendedInputCapabilities->SetCapabilities( 
            capabilities );
        }
        
    MfneSize();
    }

EXPORT_C void CEikMfne::SetUpAndDownKeysConsumed(TBool aConsume)
    {
    iFlags.Assign( EConsumeUpAndDownKeysIndex, aConsume );
    }
    
EXPORT_C void CEikMfne::SetSuppressBackgroundDrawing( TBool aSuppress )
    {
    iFlags.Assign( ESuppressBackgroundDrawing, aSuppress );
    }
    
EXPORT_C TInt CEikMfne::SetFeature( TInt aFeatureId, TInt aFeatureParam )
    {
    TInt ret = KErrNone;
    
    if ( !SupportsFeature( aFeatureId ) )
        {
        ret = KErrNotSupported;
        }
    else
        {
        switch ( aFeatureId )
            {
            case EClipGcToRect:
                if ( iExtension )
                    {
                    iExtension->iClipGcToRect = aFeatureParam;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
                
            case EDisablePenInput:
                if ( iExtension )
                    {
                    iExtension->iDisablePenInput = aFeatureParam;
                    iExtension->iFingerSupport = !( iExtension->iDisablePenInput != 0 );
                    iExtension->iHighlightAll = EFalse;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            case EFingerSupport:
                if ( iExtension )
                    {
                    iExtension->iFingerSupport = ( TBool )( aFeatureParam );
                    iExtension->iDisablePenInput = !( iExtension->iFingerSupport );                    
                    iExtension->iFingerParam = aFeatureParam; 
                    iExtension->iHighlightAll = ( iExtension->iFingerSupport &&
                        aFeatureParam == EnableWithAllHighlight );                  
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            case ELaunchPenInputAutomatic:
                if ( iExtension )
                    {
                    iExtension->iLaunchPenInputAutomatic = aFeatureParam;                    
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break; 
            case EPartialScreenInput:
                if ( iExtension )
                    {
                    iExtension->iPartialScreenInput = aFeatureParam;
                    TUint caps( iExtension->iExtendedInputCapabilities->Capabilities() );
                    if ( aFeatureParam )
                        {
                        caps |= CAknExtendedInputCapabilities::EInputEditorPartialScreen;
                        }
                    else
                        {
                        caps &= ~CAknExtendedInputCapabilities::EInputEditorPartialScreen;
                        }
                    iExtension->iExtendedInputCapabilities->SetCapabilities( caps );
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            default:
                ret = KErrNotSupported;
                break;
            }
        }
    
    return ret;
    }

EXPORT_C TInt CEikMfne::GetFeature( TInt aFeatureId, TInt& aFeatureParam ) const
    {
    TInt ret = KErrNone;
    
    if ( !SupportsFeature( aFeatureId ) )
        {
        ret = KErrNotSupported;
        }
    else
        {
        switch ( aFeatureId )
            {
            case EClipGcToRect:
                if ( iExtension )
                    {
                    aFeatureParam = iExtension->iClipGcToRect;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
                
            case EDisablePenInput:
                if ( iExtension )
                    {
                    aFeatureParam = iExtension->iDisablePenInput;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            case EFingerSupport:
                if ( iExtension )
                    {
                    aFeatureParam = iExtension->iFingerParam;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            case ELaunchPenInputAutomatic:
                if ( iExtension )
                    {
                    aFeatureParam = iExtension->iLaunchPenInputAutomatic;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            case EPartialScreenInput:
                if ( iExtension )
                    {
                    aFeatureParam = iExtension->iPartialScreenInput;
                    }
                else
                    {
                    ret = KErrGeneral;
                    }
                break;
            default:
                ret = KErrNotSupported;
                break;
            }
        }
    
    return ret;
    }

EXPORT_C TBool CEikMfne::SupportsFeature( TInt aFeatureId ) const
    {
    // This is done so that there is an option of leaving out
    // a feature instead of using the enum TFeatureId, although
    // for simplified BC that will probably never be done.
    const TInt supportedFeatures[] = 
        {
        EClipGcToRect,
        EDisablePenInput,
        EFingerSupport,
        ELaunchPenInputAutomatic,
        EPartialScreenInput
        };
        
    TBool ret = EFalse;
    
    for ( TInt i = 0; i < sizeof( supportedFeatures ) / sizeof( TInt ); ++i )
        {
        if ( supportedFeatures[i] == aFeatureId )
            {
            ret = ETrue;
            break;
            }
        }
         
    return ret;   
    }
    
void CEikMfne::HighlightField( TInt aFieldPosition )
    {
    TBool dataAltered = ETrue;
    TBool error = EFalse;
    const CFont& font=*Font();
    HandleInteraction(ETrue, aFieldPosition, iFields[iCurrentField]->WidthInPixels(font), iFields[iCurrentField]->HighlightType(), dataAltered, error);
    }

EXPORT_C const CFont* CEikMfne::Font() const
    {
    if (iFont)
        {
        return iFont;
        }
    else
        {
        return AknLayoutUtils::FontFromId( EAknLogicalFontPrimaryFont );
        }
    }

EXPORT_C void CEikMfne::SetFont(const CFont* aFont)
    {
    TBool fontChanged(iFont==aFont);
    iFont = aFont;
    if (fontChanged)
        MfneSize();
    }

EXPORT_C void CEikMfne::SetSkinBackgroundControlContextL( MAknsControlContext* aControlContext )
    {
    CreateExtensionIfRequiredL();
    if ( iExtension )
        {
        iExtension->iExternalSkinControlContext = aControlContext;
        iFlags.Set(ESkinBackGroundControlContextSetByApi);
        }
    }

EXPORT_C TSize CEikMfne::MfneSize() const
    {
    return ((CEikMfne*)this)->MfneSize(EFalse); // cast away the const-ness
    }

EXPORT_C TSize CEikMfne::MfneSize(TBool aShrinkToMinimumSize)
    {
    const CFont& font=*Font();
    TSize size=iBorder.SizeDelta(); // Shrink to border first

    size.iHeight=Rect().Height(); // Do not change height
    iBorder.SizeDelta(); // Shrink to border first

    TInt leftHighlightExtension;
    TInt rightHighlightExtension;
    TAknTextDecorationMetrics decoration( &font );
    decoration.GetLeftAndRightMargins( leftHighlightExtension, rightHighlightExtension);

    size.iWidth += leftHighlightExtension + rightHighlightExtension;

    if( iFields )
        {
        for (TInt i=0; i<iNumFields; ++i)
            size.iWidth+=iFields[i]->MaximumWidthInPixels(font, aShrinkToMinimumSize);
        }

    TRect rect = Rect();
    switch ( iAlignment )
        {
        case ELayoutAlignCenter:
            {
            TInt fieldXPosition = rect.iTl.iX + (rect.Width() - size.iWidth) / 2;
            SetExtent(TPoint(fieldXPosition, rect.iTl.iY), TSize(size));
            }
            break;
        case ELayoutAlignRight:
            {
            TInt fieldXPosition = rect.iTl.iX + (rect.Width() - size.iWidth);
            SetExtent(TPoint(fieldXPosition, rect.iTl.iY), TSize(size));
            }
            break;
        case ELayoutAlignLeft: // Left alignment is default
        case ELayoutAlignBidi: // Bidi should be ignored, so it's the same as default
        default:
            {
            iSize=size;
            }
            break;
        }

    return iSize;
    }

EXPORT_C void CEikMfne::Draw(const TRect& /*aRect*/) const
    {
    CWindowGc& gc=PreparedGc();
    TRect rect=Rect();

    // Inhibit border if skinning
    if ( !SkinningBackground() )
        iBorder.Draw(gc, rect);

    DrawRange( gc, -1, iNumFields, ETrue );
    gc.DiscardFont(); // this is not nice if there are eg. child controls to draw..

    }

EXPORT_C void* CEikMfne::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikMfne::HandlePointerEventL(const TPointerEvent& aPointerEvent)
    {
    CCoeControl::HandlePointerEventL(aPointerEvent);

    if ((aPointerEvent.iType==TPointerEvent::EButton1Down) || (aPointerEvent.iType==TPointerEvent::EDrag))
        {
        const CFont& font=*Font();
        TInt newField=ENullIndex;
        TMargins borderMargins=iBorder.Margins();
        
        TInt leftHighlightExtension;
        TInt rightHighlightExtension;
        TAknTextDecorationMetrics decoration( &font );
        decoration.GetLeftAndRightMargins( leftHighlightExtension, rightHighlightExtension );
        
        TInt leftPositionOfThisField = 
            iPosition.iX + borderMargins.iLeft + EGapLeftOfFirstField + leftHighlightExtension;
        
        // No need to do anything in the block below in case of left align
        if ( iAlignment == ELayoutAlignCenter ||
             iAlignment == ELayoutAlignRight )
            {
            TInt mfneWidth( 0 );
            for ( TInt i=0; i < iNumFields; ++i )
                {
                mfneWidth += iFields[i]->WidthInPixels( font );
                }
            
            if ( iAlignment == ELayoutAlignCenter )
                {
                leftPositionOfThisField += ( iSize.iWidth - mfneWidth ) / 2;
                }
            else if ( iAlignment == ELayoutAlignRight )
                {
                leftPositionOfThisField += ( iSize.iWidth - mfneWidth );
                }
            }
        
        // TInt leftPositionOfNewField=0; // dummy initialization to prevent compiler warning (NOT USED)
        TInt rightPositionOfNewField=0; // dummy initialization to prevent compiler warning
        for (TInt i=0; i<iNumFields; ++i)
            {
            TInt widthOfThisFieldInPixels=iFields[i]->WidthInPixels(font);
            if (iFields[i]->IsEditable())
                {
                TInt rightPositionOfThisField=(leftPositionOfThisField+widthOfThisFieldInPixels)-1;
                if (rightPositionOfThisField<aPointerEvent.iPosition.iX)
                    {
                    newField=i;
                    //leftPositionOfNewField=leftPositionOfThisField;
                    rightPositionOfNewField=rightPositionOfThisField;
                    }
                else
                    {
                    if ((newField==ENullIndex) || (leftPositionOfThisField-aPointerEvent.iPosition.iX<aPointerEvent.iPosition.iX-rightPositionOfNewField))
                        {
                        newField=i;
                        // leftPositionOfNewField and rightPositionOfNewField do not need to be set as they are not going to be used again
                        }
                    break;
                    }
                }
            leftPositionOfThisField+=widthOfThisFieldInPixels;
            }
        if (newField==ENullIndex)
            newField=iCurrentField;

        if (iExtension && aPointerEvent.iType == TPointerEvent::EButton1Down)
            {
            // Edit feedback on down event in current field, 
            // Edit feedback when changing fields
            TTouchLogicalFeedback feedback = ETouchFeedbackEdit;
            iExtension->iFeedback->InstantFeedback( this, feedback );
            }
        TBool createPopoutIfRequired=((newField==iCurrentField) && aPointerEvent.iType==TPointerEvent::EButton1Down && IsFocused());
        TBool dataAltered=EFalse;
        TBool error=EFalse;
        HandleInteraction(ETrue, newField, iFields[iCurrentField]->WidthInPixels(font), iFields[iCurrentField]->HighlightType(), dataAltered, error);
        if (createPopoutIfRequired)
            CreatePopoutIfRequiredL();
        
        ReportEventL( MCoeControlObserver::EEventStateChanged );
        ReportUpdate();
        }
    else if ( aPointerEvent.iType == TPointerEvent::EButton1Up )
        {
        if ( iExtension && !iExtension->iDisablePenInput )
            {
            iExtension->iFeedback->InstantFeedback( this,
                ETouchFeedbackEdit,
                ETouchFeedbackVibra,
                aPointerEvent );
            LaunchPenInputL();
            }
        }
    }
    
EXPORT_C TTypeUid::Ptr CEikMfne::MopSupplyObject( TTypeUid aId )
    {
    if ( aId.iUid == CAknExtendedInputCapabilities::ETypeId && iExtension )
        {
        return aId.MakePtr( iExtension->iExtendedInputCapabilities );
        }
    
    return CCoeControl::MopSupplyObject( aId );
    }

EXPORT_C void CEikMfne::CreatePopoutIfRequiredL()
    {
    // does nothing
    }

// ----------------------------------------------------------------------------
// Draws the specified range of fields.
// ----------------------------------------------------------------------------
//
void CEikMfne::DrawRange( CWindowGc& aGc,
                          TInt aFirstField,
                          TInt aLastField,
                          TBool aSkipBackgroundDrawer ) const
    {
    const MCoeControlBackground* backgroundDrawer = FindBackground();

    TInt leftHighlightExtension;
    TInt rightHighlightExtension;

    TInt topHighlightExtension;
    TInt bottomHighlightExtension;
    TAknTextDecorationMetrics decoration( Font() );
    decoration.GetTopAndBottomMargins( topHighlightExtension, bottomHighlightExtension);
    decoration.GetLeftAndRightMargins( leftHighlightExtension, rightHighlightExtension);

    // Some values that will be needed for Skins.  These will behave when not skinned, etc..
    MAknsSkinInstance* skin = AknsUtils::SkinInstance();
    MAknsControlContext* edCc = SkinBackgroundControlContext();

    TInt skinDrawFlags = KAknsDrawParamDefault;
    if ( CAknEnv::Static()->TransparencyEnabled() )
        {
        skinDrawFlags = KAknsDrawParamNoClearUnderImage;
        }

    // N.B. a value of iNumFields for aFirstField or aLastField is permitted, this means draw the space after the last field, and similarly
    // a value of -1 is permitted which means draw the space before the first field
    __ASSERT_DEBUG((aFirstField>=-1) && (aFirstField<=iNumFields) && (aLastField>=-1) && (aLastField<=iNumFields), Panic(EEikPanicMfneBadRangeToDraw));
    // N.B. aLastField is permitted to be less than aFirstField, in which case nothing will be drawn
    if (aFirstField<=aLastField)
        {
        TBool drawtrailingSpace=EFalse;
        if (aLastField==iNumFields)
            {
            if (iAlignment != ELayoutAlignRight)
                drawtrailingSpace=ETrue;
            --aLastField;
            }
        TMargins borderMargins=iBorder.Margins();
        
        TInt alignLeftOffset = VerticalOffsetForMfnePosition();
        const CFont& font=*Font();
        
        TBool clipGcToRect( EFalse );
        
        if ( iExtension )
            {
            clipGcToRect = iExtension->iClipGcToRect;
            }
                
        if (iAlignment == ELayoutAlignRight)
            {
            alignLeftOffset = iSize.iWidth-VerticalOffsetForMfnePosition();
            for (TInt i = 0; i <= aLastField;i++ )
                alignLeftOffset -= iFields[i]->WidthInPixels(font);
            
            if (alignLeftOffset < 0 && !clipGcToRect )
                alignLeftOffset = 0; // fields are not goin to fit anyway.
            }
        if (iAlignment == ELayoutAlignCenter)
            {
            alignLeftOffset = iSize.iWidth-VerticalOffsetForMfnePosition();
            for (TInt i = 0; i <= aLastField;i++ )
                alignLeftOffset -= iFields[i]->WidthInPixels(font);
            
            if (alignLeftOffset < 0 && !clipGcToRect )
                alignLeftOffset = 0; // fields are not goin to fit anyway.
            alignLeftOffset /= 2;
            }
        
        const TPoint topLeftOfMfne(iPosition.iX+borderMargins.iLeft, iPosition.iY+borderMargins.iTop);
        TPoint topLeftOfField(topLeftOfMfne);
        topLeftOfField.iX += alignLeftOffset;
		
        TInt posX = topLeftOfField.iX; 
        TRect controlRect = Rect();
        posX += leftHighlightExtension;
        for (TInt i = 0; i <= aLastField;i++ )
            {
            posX += iFields[i]->WidthInPixels(font); 
            }
        if (posX < controlRect.iBr.iX )
            {
            topLeftOfField.iX+= leftHighlightExtension;
            }

        TInt textHeight=font.HeightInPixels();
    
        if (aFirstField==-1)
            {
            SetGcToNormalVideo(aGc);
            // Clear the area between the mfne left border and the first field of the mfne
            TRect clearRect( topLeftOfMfne, TSize( (topLeftOfField.iX - topLeftOfMfne.iX), Rect().Height() ) );

            TBool skinnedDraw = EFalse;

            if ( !iFlags[ESuppressBackgroundDrawing] )
                {
                if ( backgroundDrawer )
                    {
                    if ( !aSkipBackgroundDrawer )
                        {
                        backgroundDrawer->Draw( aGc, *this, clearRect );
                        }

                    skinnedDraw = ETrue;
                    }
                else if ( SkinningBackground() && !iFlags[EUseOverrideColors] )
                    {
                    skinnedDraw = AknsDrawUtils::Background( 
                        skin, edCc, this, aGc, clearRect, skinDrawFlags );
                    }
                if ( !skinnedDraw ) // legacy code ( rect construction move to above)
                    {
                    aGc.Clear( clearRect );
                    }
                }

            ++aFirstField;
            }
        //if there is one field with text, whole MFNE editor should 
        //be highlighted when highlight all feature is enabled.
        TBool isEmpty( ETrue );
        for ( TInt i( 0 ); i< aLastField; i++ )
            {
            if ( iFields[i]->FieldText().Length() > 0 )
                {
                isEmpty = EFalse;
                break;
                }
            }       
        TBool focused = IsFocused();
        for (TInt i=0; i<=aLastField; ++i)
            {
            if (i>=aFirstField)
                {
                if (focused && iExtension && ( iExtension->iHighlightAll || ( i == iCurrentField 
                    && iFields[i]->HighlightType() == CEikMfneField::EInverseVideo
                    && iFields[i]->FieldText().Length() > 0 ) ) && !isEmpty )
                    {
                    // Currently no skin effect for the highlighted field drawing
                    SetGcToInverseVideo(aGc);
                    }
                else 
                    {
                    if (IsDimmed())
                        SetGcToDimmedVideo(aGc);
                    else
                        SetGcToNormalVideo(aGc);

                    // Note!! This is taking responsibility away from the CEikMfne Field for drawing
                    // Skin background is drawn for the fields here
                    if( SkinningBackground() && !iFlags[EUseOverrideColors] )
                        {
                        // Note that in case EUseOverrideColors is up, there is no need
                        // to clear here, as the ENullBrush below is skipped and therefore
                        // the field will draw its own background with the overridden color
                        if ( !iFlags[ESuppressBackgroundDrawing] )
                            {
                            TRect clearRect(
                                TPoint( topLeftOfField.iX, topLeftOfMfne.iY ),
                                TPoint( 
                                        topLeftOfField.iX +(iFields[i]->WidthInPixels(font)), 
                                        topLeftOfMfne.iY + Rect().Height() 
                                        )  );

                            if ( backgroundDrawer )
                                {
                                if ( !aSkipBackgroundDrawer )
                                    {
                                    backgroundDrawer->Draw( aGc, *this, clearRect );
                                    }
                                }
                            else
							    {
                                AknsDrawUtils::Background( 
                                    skin, edCc, this, aGc, clearRect, skinDrawFlags );
                                }
                            }

                        // Set up null brush GC here for the upcoming draw
                        aGc.SetBrushStyle(CGraphicsContext::ENullBrush);
                        }
                    }
                // This draw must only draw the text with a null brush or draw block highlight 
                iFields[i]->Draw(aGc, font, topLeftOfField);
                }
            topLeftOfField.iX+=iFields[i]->WidthInPixels(font); // Increment field Position
            }
        if (drawtrailingSpace)
            {
            SetGcToNormalVideo(aGc);
            if ( !iFlags[ESuppressBackgroundDrawing] )
                {
                TInt width=0;
                if (iAlignment == ELayoutAlignCenter)
                {
                width = iSize.iWidth-VerticalOffsetForMfnePosition();
                for(TInt i=0;i<iNumFields;i++)
                    width -= iFields[i]->WidthInPixels(font);
                if (width < 0) width = 0;
                width /= 2;
                }
                
                TRect clearRect;
                if ( CAknEnv::Static()->TransparencyEnabled() )
                    {
                    clearRect = TRect( topLeftOfField,
                                    TPoint( iPosition.iX + iSize.iWidth, topLeftOfMfne.iY+Rect().Height() ));
                    }
                else
                    {
                    clearRect = TRect( topLeftOfField,
                                    TPoint(iPosition.iX + width + iSize.iWidth, topLeftOfMfne.iY+Rect().Height() ));
                    }

                TBool skinnedDraw = EFalse;

                if ( backgroundDrawer )
                    {
                    if ( !aSkipBackgroundDrawer )
                        {
                        backgroundDrawer->Draw( aGc, *this, clearRect );
                        }

                    skinnedDraw = ETrue;
                    }
                else if ( SkinningBackground() && !iFlags[EUseOverrideColors] )
                    {
                    skinnedDraw = AknsDrawUtils::Background( 
                        skin, edCc, this, aGc, clearRect, skinDrawFlags );
                    }
                if ( !skinnedDraw )
                    {
                    aGc.Clear( clearRect );
                    }
                }
            }
        }
    }

CWindowGc& CEikMfne::PreparedGc() const
    {
    CWindowGc& gc=SystemGc();
    gc.SetBrushStyle(CWindowGc::ESolidBrush);
    gc.UseFont(Font()); // The font must be discarded also -> always when using PreparedGc(), 
    
    if ( iExtension && iExtension->iClipGcToRect )
        {
        gc.SetClippingRect( Rect() );
        }
    
    return gc;          // and finishing drawing, gc.DiscardFont must be called
    }

void CEikMfne::SetGcToNormalVideo(CWindowGc& aGc) const
    {
    TRgb textColor = AKN_LAF_COLOR(215);
    TRgb bgColor = iEikonEnv->ControlColor(EColorControlBackground,*this);

    MAknsSkinInstance* skin = AknsUtils::SkinInstance();

    if ( iFlags[EUseOverrideColors] )
        {
        textColor = iEikonEnv->ControlColor( EColorControlText, *this );
        }
    else if ( skin && iExtension )
       {
       AknsUtils::GetCachedColor(skin, textColor, KAknsIIDQsnTextColors, iExtension->iAknSkinColorIndex); 
       AknsUtils::GetCachedColor(skin, bgColor, KAknsIIDQsnTextColors, iExtension->iSkinIdForTextBackgroundColor);         
       }
        
    aGc.SetPenColor(textColor);  // Text color
    aGc.SetBrushColor(bgColor); 
    }

void CEikMfne::SetGcToInverseVideo(CWindowGc& aGc) const
    {
    TRgb hightext = AKN_LAF_COLOR(0); // Hightlighted text color   // AVKON LAF
    TRgb highcolor = AKN_LAF_COLOR(210); // Highlight color  // AVKON LAF

    if ( iFlags[EUseOverrideColors] )
        {
        hightext = iEikonEnv->ControlColor( EColorControlHighlightText, *this );
        highcolor = iEikonEnv->ControlColor( EColorControlHighlightBackground, *this );
        }
    else
        {
        MAknsSkinInstance* skin = AknsUtils::SkinInstance();
        AknsUtils::GetCachedColor(skin, hightext, KAknsIIDQsnTextColors, EAknsCIQsnTextColorsCG24);
        AknsUtils::GetCachedColor(skin, highcolor, KAknsIIDQsnHighlightColors, EAknsCIQsnHighlightColorsCG2);
        }

    aGc.SetPenColor(hightext); 
    aGc.SetBrushColor(highcolor); 

    // Ensure highlight in inverse video
    aGc.SetBrushStyle(CGraphicsContext::ESolidBrush);
    }

void CEikMfne::SetGcToDimmedVideo(CWindowGc& aGc) const
    {
    if ( iFlags[EUseOverrideColors] )
        {
        aGc.SetPenColor(iEikonEnv->ControlColor(EColorControlDimmedText,*this)); //KEikMfneDimmedColor);
        aGc.SetBrushColor(iEikonEnv->ControlColor(EColorControlBackground,*this)); //KEikMfneBackgroundColor);
        }
    else
        {
        SetGcToNormalVideo(aGc);
        }
    }

void CEikMfne::GetCursorInfo( TPoint& aPos, TInt& aHeight, TInt& aWidth, 
    TInt& aAscent )
    {
    const CFont& font=*Font();
    const CAknLayoutFont* layoutFont = CAknLayoutFont::AsCAknLayoutFontOrNull( &font );
    if ( layoutFont )
        {
        TInt topMargin;
        TInt bottomMargin;
        TAknTextDecorationMetrics metrics = layoutFont->TextDecorationMetrics();
        metrics.GetTopAndBottomMargins( topMargin, bottomMargin );
        aHeight = layoutFont->TextPaneHeight() + topMargin + bottomMargin;
        aWidth  = metrics.CursorWidth();
        aAscent = layoutFont->TextPaneHeight();
        // The ascent is according to CursorAscentFromFont. It doesn't really make a
        // difference because it is added to the cursor TPoint and then used again
        // in CEikonEnv::DrawCursor
        }
    else
        { // The old way
        aHeight = AknLayoutUtils::CursorHeightFromFont( font.FontSpecInTwips() );
        aWidth  = AknLayoutUtils::CursorWidthFromFont ( font.FontSpecInTwips() );
        aAscent = AknLayoutUtils::CursorAscentFromFont( font.FontSpecInTwips() );
        }

    TMargins borderMargins = iBorder.Margins();
    aPos.iX = iPosition.iX + borderMargins.iLeft + EGapLeftOfFirstField;
    aPos.iY = iPosition.iY + borderMargins.iTop + EGapAboveText + 
        aAscent + ( EMfneFieldExtraHeight / 2 );
    
    if ( iAlignment == ELayoutAlignRight )
        {
        aPos.iX = iPosition.iX + iSize.iWidth - VerticalOffsetForMfnePosition();
        for ( TInt i = iNumFields-1 ; i >= iCurrentField ; i-- )
            {
            aPos.iX -= iFields[i]->WidthInPixels(font);
            }           
        }
    else if (iAlignment == ELayoutAlignCenter)
        {
        TInt width = iSize.iWidth-VerticalOffsetForMfnePosition();
        for ( TInt i = 0; i < iNumFields; i++ )
            {
            width -= iFields[i]->WidthInPixels( font );
            }           
        if ( width < 0 )
            {
            width = 0;
            }
        width /= 2;
        aPos.iX += width;
        for (TInt i = 0; i<iCurrentField; ++i)
            {
            aPos.iX += iFields[i]->WidthInPixels( font );       
            }           
        }
    else
        {
        for ( TInt i = 0; i < iCurrentField; ++i )
            {
            aPos.iX += iFields[i]->WidthInPixels( font );
            }           
        }
        
    aPos.iX += iFields[iCurrentField]->
        DistanceFromStartOfFieldToEndOfTextInPixels( font );
    }

void CEikMfne::DrawCursor()
    {
    __ASSERT_DEBUG(IsFocused() && (iFields[iCurrentField]->HighlightType()==CEikMfneField::ECursor), Panic(EEikPanicMfneBadCursorState3));  

    TInt cursorHeight( 0 );
    TInt cursorWidth( 0 );
    TInt cursorAscent( 0 );
    TPoint cursorPosition( 0, 0 );

    GetCursorInfo( cursorPosition, cursorHeight, cursorWidth, cursorAscent );

    iEikonEnv->DrawCursor(this, cursorPosition, cursorWidth, cursorAscent, cursorHeight);
    if ( iExtension )
    	{
        iExtension->iCursorShown = ETrue;
    	}
    }

void CEikMfne::HideCursor()
    {
    if ( iExtension && iExtension->iCursorShown )
        {
        iEikonEnv->HideCursor(this);
        iExtension->iCursorShown = EFalse;
        }   
    }


void CEikMfne::ReportUpdate()
    {
    if ( iExtension )
        {
        TRAP_IGNORE (
            iExtension->iExtendedInputCapabilities->ReportEventL(
            CAknExtendedInputCapabilities::MAknEventObserver::EControlContentUpdatedInternally,
            NULL );
            )
        }
    }

/**
 * Gets the list of logical colors employed in the drawing of the control,
 * paired with an explanation of how they are used. Appends the list to aColorUseList.
 *
 * @since ER5U
 */
EXPORT_C void CEikMfne::GetColorUseListL(CArrayFix<TCoeColorUse>& aColorUseList) const
    {
    CEikBorderedControl::GetColorUseListL(aColorUseList);

    TInt commonAttributes = TCoeColorUse::EContents|TCoeColorUse::ENormal|TCoeColorUse::ENeutral;
    TCoeColorUse colorUse;

    colorUse.SetLogicalColor(EColorControlText);
    colorUse.SetUse(TCoeColorUse::EFore|TCoeColorUse::EActive|commonAttributes);
    aColorUseList.AppendL(colorUse);

    colorUse.SetLogicalColor(EColorControlHighlightText);
    colorUse.SetUse(TCoeColorUse::EFore|TCoeColorUse::EHighlights|commonAttributes);
    aColorUseList.AppendL(colorUse);

    colorUse.SetLogicalColor(EColorControlDimmedText);
    colorUse.SetUse(TCoeColorUse::EFore|TCoeColorUse::EDimmed|commonAttributes);
    aColorUseList.AppendL(colorUse);

    colorUse.SetLogicalColor(EColorControlBackground);
    colorUse.SetUse(TCoeColorUse::EBack|TCoeColorUse::EActive|commonAttributes);
    aColorUseList.AppendL(colorUse);

    colorUse.SetLogicalColor(EColorControlHighlightBackground);
    colorUse.SetUse(TCoeColorUse::EBack|TCoeColorUse::EHighlights|commonAttributes);
    aColorUseList.AppendL(colorUse);
    }

/**
 * Handles a change to the control's resources of type aType
 * which are shared across the environment, e.g. colors or fonts.
 *
 * @since ER5U
 */
EXPORT_C void CEikMfne::HandleResourceChange(TInt aType)
    {
    EvaluateSkinningBackground();
    CEikBorderedControl::HandleResourceChange(aType);
    if ( aType == KEikDynamicLayoutVariantSwitch && iFields )
        {
        if ( IsFocused() && iFields[iCurrentField]->HighlightType() ==
            CEikMfneField::ECursor )
            {
	        HideCursor();
            DrawCursor();
            }
        }
    }

/**
 * Writes the internal state of the control and its components to aStream.
 * Does nothing in release mode.
 * Designed to be overidden and base called by subclasses.
 *
 * @internal
 * @since App-Framework_6.1
 */
#ifndef _DEBUG
EXPORT_C void CEikMfne::WriteInternalStateL(RWriteStream&) const
    {}
#else
EXPORT_C void CEikMfne::WriteInternalStateL(RWriteStream& aWriteStream) const
    {
    CEikBorderedControl::WriteInternalStateL(aWriteStream);
    }
#endif

EXPORT_C void CEikMfne::Reserved_2()
    {
    }

EXPORT_C void CEikMfne::CEikMfne_Reserved()
    {
    }
    
    
// -----------------------------------------------------------------------------
// CEikMfne::HandleMfneCommandL
// Handles external MFNE commands
// -----------------------------------------------------------------------------
//
EXPORT_C void CEikMfne::HandleMfneCommandL(TInt aCommand)
    {
    if( !AknLayoutUtils::PenEnabled() )
        {
        return;
        }

    // store the original setting
    TBool original = ConsumesUpAndDownKeys();
    
    if(!original)
        {
        // We need to enable the up/down key support to simulate key events
        SetUpAndDownKeysConsumed(ETrue);
        }
    
    TKeyEvent event;
    event.iCode = 0;
    event.iScanCode = 0;
    event.iModifiers = 0;
    event.iRepeats = 0;
    
    switch(aCommand)
        {
        case MAknMfneCommandObserver::EMfneIncrementCurrentFieldValue:
            // Increment current field value by simulating internally
            // an up arrow key event 
            event.iCode = EKeyUpArrow;
            OfferKeyEventL(event, EEventKey);
        break;
        
        case MAknMfneCommandObserver::EMfneDecrementCurrentFieldValue:
            // Decrement current field value by simulating internally
            // a down arrow key event 
            event.iCode = EKeyDownArrow;
            OfferKeyEventL(event, EEventKey);
        break;
        
        default:
        break;
        }
        
    if(!original)
        {
        // restore the original setting
        SetUpAndDownKeysConsumed(original);
        }
    }
    
EXPORT_C void CEikMfne::SetUseOverrideColors( TBool aUseOverrideColors )
    {
    iFlags.Assign( EUseOverrideColors, aUseOverrideColors );
    }


TBool CEikMfne::ConsumesUpAndDownKeys() const
    {
    return iFlags[ EConsumeUpAndDownKeysIndex ];
    }

TBool CEikMfne::SkinningBackground() const
    {
    if (iExtension)
        return iExtension->iSkinIdForTextBackgroundColor == KErrNotFound && iFlags[ ESkinningBackground ]; 
    else
        return iFlags[ ESkinningBackground ];
    }

void CEikMfne::EvaluateSkinningBackground()
    {
    // Status of application skinning is latched at construction and resource change
    iFlags.Assign( ESkinningBackground, AknsUtils::AvkonSkinEnabled() );
    }

MAknsControlContext* CEikMfne::SkinBackgroundControlContext() const
    {
    MAknsControlContext* context = 0;

    if ( iExtension )
        context = iExtension->iExternalSkinControlContext;

    if (!context && !iFlags[ESkinBackGroundControlContextSetByApi] )
        {
        // Obtain the context from Object Provider
        context = AknsDrawUtils::ControlContext( this );
        }

    return context;
    }

void CEikMfne::CreateExtensionIfRequiredL()
    {
    if (!iExtension )
        {
        iExtension = CEikMfneExtension::NewL();
        iExtension->iEditor = this;
        iExtension->iExtendedInputCapabilities->SetEditorType(
            CAknExtendedInputCapabilities::EMFNEBased );
        iExtension->iExtendedInputCapabilitiesProvider->SetMopParent(
            this );
        }
    }

void CEikMfne::SetCurrentField( TInt aCurrentField )
    {
    iCurrentField = aCurrentField;
    }

void CEikMfne::SetFirstEditableField( )
	{
    TBool foundEditable = EFalse;
    for ( TInt i=0; i < iNumFields; ++i )
        {
        if ( iFields[i]->IsEditable() )
            {
            SetCurrentField(i);
            foundEditable = ETrue;
            break;
            }
        }
    if ( !foundEditable )
	    {
	    SetCurrentField( 0 );
	    }
	}
//
// CEikNumberEditor
//

EXPORT_C CEikNumberEditor::CEikNumberEditor()
    {
    __DECLARE_NAME(_S("CEikNumberEditor"));
    AKNTASHOOK_ADD( this, "CEikNumberEditor" );
    }

EXPORT_C void CEikNumberEditor::ConstructL(TInt aMinimumValue, TInt aMaximumValue, TInt aInitialValue)
    {
    CreateFieldArrayL(1);
    iNumber=CEikMfneNumber::NewL(*Font(), aMinimumValue, aMaximumValue, aInitialValue, 0);
    AddField(iNumber);
    RefreshFromLocale();
    }

void CEikNumberEditor::RefreshFromLocale()
/**
Update the editor from locale - ensures that digits are displayed using correct numerals for current device settings.
@publishedComponent
@param aFont the font of the MFNE which owns this class
@post the next time the editor is drawn it will display the number using digit type derived from the locale information.
*/
    {
    iNumber->RefreshDigitType(*Font());
    }

EXPORT_C void CEikNumberEditor::SetMinimumAndMaximum(TInt aMinimumValue, TInt aMaximumValue)
    {
    iNumber->SetMinimumAndMaximum(aMinimumValue, aMaximumValue, *Font());
    }

EXPORT_C void CEikNumberEditor::GetMinimumAndMaximum(TInt& aMinimumValue, TInt& aMaximumValue) const
    {
    iNumber->GetMinimumAndMaximum(aMinimumValue, aMaximumValue);
    }

EXPORT_C void CEikNumberEditor::SetNumber(TInt aNumber)
    {
    iNumber->SetValue(aNumber, *Font());
    TRAP_IGNORE ( ReportEventL( MCoeControlObserver::EEventStateChanged ) );
    }

EXPORT_C TInt CEikNumberEditor::Number() const
    {
    return iNumber->Value();
    }

EXPORT_C void CEikNumberEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TInt minimumValue=aResourceReader.ReadInt32();
    TInt maximumValue=aResourceReader.ReadInt32();
    ConstructL(minimumValue, maximumValue, minimumValue);
    }
    
EXPORT_C void CEikNumberEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikMfne::HandlePointerEventL(aPointerEvent); 
    }   

EXPORT_C void* CEikNumberEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikNumberEditor::CEikMfne_Reserved()
    {
    }


// CEikRangeEditor

EXPORT_C CEikRangeEditor::CEikRangeEditor()
    {
    __DECLARE_NAME(_S("CEikRangeEditor"));
    AKNTASHOOK_ADD( this, "CEikRangeEditor" );
    }

EXPORT_C void CEikRangeEditor::ConstructL(TInt aMinimumValue, TInt aMaximumValue, const SEikRange& aInitialRange, HBufC* aSeparatorText)
    {
    const CFont& font=*Font();
    CreateFieldArrayL(3);
    iLowerLimit=CEikMfneNumber::NewL(font, aMinimumValue, aMaximumValue, aInitialRange.iLowerLimit, 0);
    AddField(iLowerLimit);
    CEikMfneSeparator* separator=CEikMfneSeparator::NewL(NULL);
    AddField(separator);
    iUpperLimit=CEikMfneNumber::NewL(font, aMinimumValue, aMaximumValue, aInitialRange.iUpperLimit, 0);
    AddField(iUpperLimit);
    // do stuff that can only be done when all leaving functions have successfully been done
    separator->SetText(aSeparatorText);

    RefreshFromLocale();
    }

void CEikRangeEditor::RefreshFromLocale()
    {
    iLowerLimit->RefreshDigitType(*Font() );
    iUpperLimit->RefreshDigitType(*Font() );
    }

EXPORT_C void CEikRangeEditor::SetMinimumAndMaximum(TInt aMinimumValue, TInt aMaximumValue)
    {
    const CFont& font=*Font();
    iLowerLimit->SetMinimumAndMaximum(aMinimumValue, aMaximumValue, font);
    iUpperLimit->SetMinimumAndMaximum(aMinimumValue, aMaximumValue, font);
    }

EXPORT_C void CEikRangeEditor::GetMinimumAndMaximum(TInt& aMinimumValue, TInt& aMaximumValue) const
    {
    iLowerLimit->GetMinimumAndMaximum(aMinimumValue, aMaximumValue);
#if defined(_DEBUG)
    TInt minimumValue;
    TInt maximumValue;
    iUpperLimit->GetMinimumAndMaximum(minimumValue, maximumValue);
    __ASSERT_DEBUG((minimumValue==aMinimumValue) && (maximumValue==aMaximumValue), Panic(EEikPanicRangeEditorInconsistentMinimumAndMaximum));
#endif
    }

EXPORT_C void CEikRangeEditor::SetRange(const SEikRange& aRange)
    {
    const CFont& font=*Font();
    iLowerLimit->SetValue(aRange.iLowerLimit, font);
    iUpperLimit->SetValue(aRange.iUpperLimit, font);
    }

EXPORT_C SEikRange CEikRangeEditor::Range() const
    {
    SEikRange range;
    range.iLowerLimit=iLowerLimit->Value();
    range.iUpperLimit=iUpperLimit->Value();
    return range;
    }

EXPORT_C void CEikRangeEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TInt minimumValue=aResourceReader.ReadInt32();
    TInt maximumValue=aResourceReader.ReadInt32();
    HBufC* separatorText=aResourceReader.ReadHBufCL();
    CleanupStack::PushL(separatorText);
    SEikRange initialRange;
    initialRange.iLowerLimit=maximumValue;
    initialRange.iUpperLimit=maximumValue;
    ConstructL(minimumValue, maximumValue, initialRange, separatorText);
    CleanupStack::Pop();
    }

void CEikRangeEditor::FieldIsAboutToBeDeHighlighted(CEikMfneField* aField, TBool& aDrawAllFields)
    {
    const CFont& font=*Font();
    if (aField==iLowerLimit)
        {
        TInt lowerLimitValue=iLowerLimit->Value();
        if (iUpperLimit->Value()<lowerLimitValue)
            {
            iUpperLimit->SetValue(lowerLimitValue, font);
            aDrawAllFields=ETrue;
            }
        }
    else if (aField==iUpperLimit)
        {
        TInt upperLimitValue=iUpperLimit->Value();
        if (iLowerLimit->Value()>upperLimitValue)
            {
            iLowerLimit->SetValue(upperLimitValue, font);
            aDrawAllFields=ETrue;
            }
        }
    }
    
EXPORT_C void CEikRangeEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikMfne::HandlePointerEventL(aPointerEvent); 
    }   

EXPORT_C void* CEikRangeEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikRangeEditor::CEikMfne_Reserved()
    {
    }


// CTimeEditor

class CTimeEditor : public CBase
    {
public:
    enum
        {
        EWithoutSecondsField        =1, // same as EEikTimeWithoutSecondsField,
        EWithoutPopoutCalendar      =2, // same as EEikDateWithoutPopoutCalendar,
        EWithoutHoursField          =4, // same as EEikTimeWithoutHoursField,
        EForce24HourFormat          =8, // same as EEikTimeForce24HourFormat,
        ETimeZoneOffsetFormat       =16,// same as EEikTimeZoneOffsetFormat,
        };
public:
    CTimeEditor();
    TInt NumFieldsRequired(TUint32 aFlags);
    void ConstructFieldsL(CEikMfne& aOwner, const TTime& aInitialTime, const CFont& aFont);
    void SetTime(const TTime& aTime, const CFont& aFont);
    void SetMinMax(TInt aHourMin,TInt aHourMax,TInt aMinuteMin,TInt aMinuteMax,
        TInt aSecondMin,TInt aSecondMax,const CFont& aFont);
    TTime Time() const;
    void SetUninitialised(TBool aUninitialised);
    TBool IsUninitialised() const;
    void RefreshFromLocale(const CFont& aFont);

private:
    void AddAmPmFieldsIfNecessaryL(CEikMfne& aOwner, TAmPm aAmPm, TLocalePos aPos);
private:
    enum // these are needed in case the OS locale changes between NumFieldsRequired and ConstructFieldsL being called
        {
        EIn12HourFormat             =0x20,
        ESpaceBeforeAmPm            =0x40,
        EAmPmIsPositionedBeforeTime =0x80,
        };
private:
    TUint32 iFlags;
    // none of these pointers owns anything
    CEikMfneNumber* iSecond; // may be NULL;
    CEikMfneNumber* iMinute;
    CEikMfneNumber* iHour; // may be NULL;
    CEikMfneSymbol* iAmPm; // may be NULL;
    };

CTimeEditor::CTimeEditor()
    {
    __DECLARE_NAME(_S("CTimeEditor"));
    }

TInt CTimeEditor::NumFieldsRequired(TUint32 aFlags)
    {
    iFlags=aFlags;
    TInt numFieldsRequired=1; // 1 for minute
    if (~iFlags&EWithoutSecondsField)
        numFieldsRequired+=2; // 2 for separator and second
    if (~iFlags&EWithoutHoursField)
        numFieldsRequired+=2; // 2 for hour and separator
    if (~iFlags&EForce24HourFormat&&~iFlags&ETimeZoneOffsetFormat)
        {
        TLocale locale;
        TTimeFormat timeFormat=locale.TimeFormat();
        __ASSERT_DEBUG((timeFormat==ETime12) || (timeFormat==ETime24), Panic(EEikPanicDateEditorBadTimeFormat));
        if (timeFormat==ETime12)
            {
            if (locale.AmPmSymbolPosition()==ELocaleBefore)
                iFlags|=EAmPmIsPositionedBeforeTime;
            if (locale.AmPmSpaceBetween())
                {
                ++numFieldsRequired;
                iFlags|=ESpaceBeforeAmPm;
                }
            ++numFieldsRequired;
            iFlags|=EIn12HourFormat;
            }
        }
    return numFieldsRequired;
    }

void CTimeEditor::RefreshFromLocale(const CFont& aFont)
/**
Update the tier editor from locale - ensures that digits are displayed using correct numerals for current device settings.
@publishedComponent
@param aFont the font of the MFNE which owns this class
@post the next time the editor is drawn it will display the number using digit type derived from the locale information.
*/
    {
    if(iSecond)
        iSecond->RefreshDigitType(aFont);

    if(iMinute)
        iMinute->RefreshDigitType(aFont);

    if(iHour)
        iHour->RefreshDigitType(aFont);
    }

void CTimeEditor::ConstructFieldsL(CEikMfne& aOwner, const TTime& aInitialTime, const CFont& aFont)
    {
    TDateTime initialTime=aInitialTime.DateTime();
    TLocale locale;
    TInt timeSeparatorIndex=1;
    const TUint32 flags=CEikMfneNumber::EFillWithLeadingZeros|CEikMfneNumber::EPreserveOldWidthBeforeEditing;
    TInt hour=initialTime.Hour();
    TAmPm amPm=(hour<12)? EAm: EPm;
    TInt minimumHour=0;
    TInt maximumHour=23;
    if (iFlags&EIn12HourFormat)
        {
        minimumHour=1;
        maximumHour=12;
        hour%=12;
        if (hour==0)
            hour=12;
        }
    if (iFlags&ETimeZoneOffsetFormat)
        {
        minimumHour=0;
        maximumHour=12;
        hour=0;
        }
    // create and push the time elements in the opposite order to the order in which they will be added so that they are popped off in the correct order
    AddAmPmFieldsIfNecessaryL(aOwner, amPm, ELocaleBefore);
    TBool amIsBefor(iFlags&EAmPmIsPositionedBeforeTime);
    if (~iFlags&EWithoutSecondsField)
        {
        iSecond=CEikMfneNumber::NewL(aFont, 0, KMaxMinSecValue, initialTime.Second(), flags);
        CleanupStack::PushL(iSecond); // this needs to be pushed as it would not be destroyed by the class' destructor
        }
    iMinute=CEikMfneNumber::NewL(aFont, 0, KMaxMinSecValue, initialTime.Minute(), flags);
    CleanupStack::PushL(iMinute); // this needs to be pushed as it would not be destroyed by the class' destructor
    if (~iFlags&EWithoutHoursField)
        {
        //iHour=CEikMfneNumber::NewL(aFont, minimumHour, maximumHour, hour, flags&~((iFlags&EIn12HourFormat)? CEikMfneNumber::EFillWithLeadingZeros: 0));
        iHour=CEikMfneNumber::NewL(aFont, minimumHour, maximumHour, hour,flags);
        CleanupStack::PushL(iHour); // this needs to be pushed as it would not be destroyed by the class' destructor
        // add hour
        aOwner.AddField(iHour);
        CleanupStack::Pop();
        // add the first separator
        HBufC* firstSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
        firstSeparatorText->Des().Append(locale.TimeSeparator(timeSeparatorIndex++));
        aOwner.AddField(CEikMfneSeparator::NewL(firstSeparatorText));
        CleanupStack::Pop();
        }
    // add minute
    aOwner.AddField(iMinute);
    CleanupStack::Pop();
    if (~iFlags&EWithoutSecondsField)
        {
        // add the second separator
        HBufC* secondSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
        secondSeparatorText->Des().Append(locale.TimeSeparator(timeSeparatorIndex++));
        aOwner.AddField(CEikMfneSeparator::NewL(secondSeparatorText));
        CleanupStack::Pop();
        // add second
        aOwner.AddField(iSecond);
        CleanupStack::Pop();
        }
    if ( !amIsBefor )
        {
        AddAmPmFieldsIfNecessaryL(aOwner, amPm, ELocaleAfter);    
        }    
    else
        {
        // if am is added before time fields, set first editable time field 
        // as highlighted        
        for ( TInt i = 1; aOwner.Field( i ) != NULL; i++ )
            {
            if ( aOwner.Field( i )->IsEditable() )
                {
                aOwner.SetCurrentField( i );
                break;
                }                
            }        
        }    
    }

void CTimeEditor::SetTime(const TTime& aTime, const CFont& aFont)
    {
    TDateTime time=aTime.DateTime();
    if (iSecond)
        iSecond->SetValue(time.Second(), aFont);
    iMinute->SetValue(time.Minute(), aFont);
    TInt hour=time.Hour();
    if (iAmPm)
        {
        TAmPm amPm=(hour<12)? EAm: EPm;
        hour%=12;
        if (hour==0)
            hour=12;
        iAmPm->SetCurrentSymbolicItemToId(amPm);
        }
    if(iHour)
        iHour->SetValue(hour, aFont);
    }

void CTimeEditor::SetMinMax(TInt /*aHourMin*/,TInt /*aHourMax*/,TInt /*aMinuteMin*/,TInt /*aMinuteMax*/,
        TInt /*aSecondMin*/,TInt /*aSecondMax*/,const CFont& /*aFont*/)
    {
    /*iHour->SetMinimumAndMaximum(aHourMin, aHourMax, aFont);
    iMinute->SetMinimumAndMaximum(aMinuteMin, aMinuteMax, aFont);
    if (iSecond!=NULL)
        iSecond->SetMinimumAndMaximum(aSecondMin, aSecondMax, aFont);*/
    }

TTime CTimeEditor::Time() const
    {
    TInt dayValue=0;
    TInt hourValue=iHour?iHour->Value():0;
    if (iAmPm)
        {
        hourValue%=12;
        if (iAmPm->IdOfCurrentSymbolicItem()==EPm)
            hourValue+=12;
        }
    return TTime(TDateTime(0, EJanuary, dayValue, hourValue, iMinute->Value(), iSecond? iSecond->Value(): 0, 0));
    }

void CTimeEditor::AddAmPmFieldsIfNecessaryL(CEikMfne& aOwner, TAmPm aAmPm, TLocalePos aPos)
    {
    if ( iFlags&EIn12HourFormat )
        {
        // Test the am/pm symbol 
        TAmPmName amName(EAm);
        TAmPmName pmName(EPm);

        TBool stackedBefore = iFlags&EAmPmIsPositionedBeforeTime;

        // Use the directionality of the first character of the localized Am symbol
        // to determine if "Before" means stacked first or last. 
        TChar firstChar('A') ;
        if (amName.Length()>0)
            {
            firstChar = amName[0];
            TInt cat = firstChar.GetBdCategory();
            // RTL indicates that we have to put it stack as if after.
            if ( cat == TChar::ERightToLeft || cat == TChar::ERightToLeftArabic )
                stackedBefore = !(iFlags&EAmPmIsPositionedBeforeTime);
            }

        if ( ( stackedBefore && ( aPos==ELocaleBefore ) )
            || ( !stackedBefore && (aPos!=ELocaleBefore ) ) )
            {
            CEikMfneSeparator* space=NULL;
            if (iFlags&ESpaceBeforeAmPm)
                {
                // add separator consisting of a space
                HBufC* spaceSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
                spaceSeparatorText->Des().Append(TChar(' '));
                space=CEikMfneSeparator::NewL(spaceSeparatorText);
                CleanupStack::Pop();
                CleanupStack::PushL(space);
                }
            iAmPm=CEikMfneSymbol::NewL(2);
            CleanupStack::PushL(iAmPm); // this needs to be pushed as it would not be destroyed by the class' destructor

            // add the am symbol
            HBufC* amText=HBufC::NewLC(amName.Length()); // also pushes on to CleanupStack
            *amText=amName;
            iAmPm->AddSymbolicItem(CEikMfneSymbol::CItem::NewL(EAm, amName[0], amText), aAmPm==EAm); // !! IS amName[0] SUFFICIENT FOR THE KEY TO MATCH?
            CleanupStack::Pop();
            // add the pm symbol
            HBufC* pmText=HBufC::NewLC(pmName.Length()); // also pushes on to CleanupStack
            *pmText=pmName;
            iAmPm->AddSymbolicItem(CEikMfneSymbol::CItem::NewL(EPm, pmName[0], pmText), aAmPm==EPm); // !! IS pmName[0] SUFFICIENT FOR THE KEY TO MATCH?
            CleanupStack::Pop((iFlags&ESpaceBeforeAmPm)? 3: 2);
            if ( (!stackedBefore) && space )
                aOwner.AddField(space);
            aOwner.AddField(iAmPm);
            if ( stackedBefore && space )
                aOwner.AddField(space);
            }
        }
    }

void CTimeEditor::SetUninitialised(TBool aUninitialised)
    {
    if (iSecond)
        iSecond->SetUninitialised(aUninitialised);
    if (iMinute)
        iMinute->SetUninitialised(aUninitialised);
    if (iHour)
        iHour->SetUninitialised(aUninitialised);
    if (iAmPm)
        iAmPm->SetUninitialised(aUninitialised);
    }

TBool CTimeEditor::IsUninitialised() const
    {
    return 
        (iSecond && iSecond->IsUninitialised()) ||
        (iMinute && iMinute->IsUninitialised()) ||
        (iHour && iHour->IsUninitialised()) ||
        (iAmPm && iAmPm->IsUninitialised());
    }


// CDateEditor

class CDateEditor : public CBase
    {
public:
    CDateEditor(CEikMfne& aOwner, MEikCalendarObserver* aCalendarObserver, TInt aFirstFieldPosition);
    TInt NumFieldsRequired() const;
    void ConstructFieldsL(TInt aMinimumYear, TInt aMaximumYear, const TTime& aInitialDate, TBool aWithoutPopoutCalendar, CEikonEnv& aEikonEnv, const CFont& aFont);
    void SetMinimumAndMaximum(TInt aMinimumYear, TInt aMaximumYear, const CFont& aFont);
    void SetDate(const TTime& aDate, const CFont& aFont);
    TTime Date(const TTime* aTime=NULL) const;
    TKeyResponse OfferKeyEventL(const TKeyEvent& aKeyEvent, TEventCode aType);
    void PrepareForFocusLossL();
    void CreatePopoutIfRequiredL();
    void ExecuteCalendarL();

    void SetUninitialised(TBool aUninitialised);
    TBool IsUninitialised() const;
    void RefreshFromLocale(const CFont& aFont);
    
    static TInt ValidateCallBack( TAny* aPtr );

public:
    TInt Year() const;
    TInt Month() const;
    TInt Day() const;
    TBool IsTextNull() const;
    void UpdateMaxDay( const CFont& aFont );
private:
    class RDateElements // objects of this class must be pushed on to the cleanup-stack
        {
    public:
        RDateElements();
        void Add(CEikMfneNumber* aDateElement, TInt aPosition);
        CEikMfneNumber* Next();
    private:
        static void Cleanup(TAny* aThis);
    public:
        inline operator TCleanupItem() {return TCleanupItem(Cleanup, this);}
    private:
        enum {EMaximumNumberOfDateElements=3};
    private:
        CEikMfneNumber* iDateElements[EMaximumNumberOfDateElements];
        };

private:
    /**
    * Called to see if the date fields need to be re-ordered because the 
    * number format causes logical to visual reordering in text.
    */
    static TBool DateFieldReOrderingNeeded();

private:
    CEikMfne& iOwner;
    MEikCalendarObserver* iCalendarObserver;
    TBool iWithoutPopoutCalendar;
    // none of these pointers owns anything
    CEikMfneNumber* iDay;
    CEikMfneNumber* iMonth;
    CEikMfneNumber* iYear;
    TInt iDayFieldPosition;
    TInt iFirstFieldPosition;
    };

CDateEditor::CDateEditor(CEikMfne& aOwner,MEikCalendarObserver* aCalendarObserver,TInt aFirstFieldPosition)
    :iOwner(aOwner),
     iCalendarObserver(aCalendarObserver),
     iFirstFieldPosition(aFirstFieldPosition)
    {
    __DECLARE_NAME(_S("CDateEditor"));
    }

TInt CDateEditor::NumFieldsRequired() const
    {
    return 5;
    }

TBool CDateEditor::DateFieldReOrderingNeeded()
    {
    TBool reorderingNeeded(EFalse);

    TLocale locale;

    // Reordering needed if it is RTL, we have arabic digits and if the separator is not 
    // numeric separator.
    TLanguage language = (TLanguage)(User::Language() & KAknLanguageMask);
    if ( TBidiText::ScriptDirectionality( language ) == TBidiText::ERightToLeft)
        {
        TInt digitType = AknTextUtils::NumericEditorDigitType();
        if ( digitType == EDigitTypeArabicIndic )
            //|| digitType == EDigitTypeEasternArabicIndic ) // Urdu has numbers bidi=EN (LTR)
                                                             // no re-ordering is needed
            {
            TChar sep = locale.DateSeparator(1); 
            if ( sep.GetBdCategory() != TChar::ECommonNumberSeparator )
                reorderingNeeded = ETrue;
            }
        }
    return reorderingNeeded;
    }

void CDateEditor::ConstructFieldsL(TInt aMinimumYear, TInt aMaximumYear, const TTime& aInitialDate, TBool aWithoutPopoutCalendar, CEikonEnv& /*aEikonEnv*/, const CFont& aFont)
    {
    iWithoutPopoutCalendar=aWithoutPopoutCalendar;
    const TUint32 flags=CEikMfneNumber::EFillWithLeadingZeros|CEikMfneNumber::EPreserveOldWidthBeforeEditing;
    TLocale locale;
    TDateFormat dateFormat=locale.DateFormat();
    TInt dateSeparatorIndex=1;
    TInt dayPosition=0, monthPosition=0, yearPosition=0; // dummy initializations to prevent compiler warnings

    // Implement switch in layout owning to locale:
    if ( DateFieldReOrderingNeeded() )
        {
        switch (dateFormat)
            {
        case EDateAmerican:
            monthPosition=2;
            dayPosition=1;
            yearPosition=0;
            break;
        case EDateEuropean:
            dayPosition=2;
            monthPosition=1;
            yearPosition=0;
            break;
        case EDateJapanese:
            yearPosition=2;
            monthPosition=1;
            dayPosition=0;
            break;
#if defined(_DEBUG)
        default:
            Panic(EEikPanicDateEditorBadDateFormat);
            break;
#endif
            }
        }
    else
        {
        switch (dateFormat)
            {
        case EDateAmerican:
            monthPosition=0;
            dayPosition=1;
            yearPosition=2;
            break;
        case EDateEuropean:
            dayPosition=0;
            monthPosition=1;
            yearPosition=2;
            break;
        case EDateJapanese:
            yearPosition=0;
            monthPosition=1;
            dayPosition=2;
            break;
#if defined(_DEBUG)
        default:
            Panic(EEikPanicDateEditorBadDateFormat);
            break;
#endif
            }
        }

    TDateTime initialDate=aInitialDate.DateTime();
    RDateElements dateElements;
    CleanupStack::PushL(dateElements);
    iDay=CEikMfneNumber::NewL(aFont, 1, 31, initialDate.Day()+1, flags);

    iDayFieldPosition = dayPosition * 2;
    dateElements.Add(iDay, dayPosition);
    iMonth=CEikMfneNumber::NewL(aFont, 1, 12, initialDate.Month()+1, flags);
    dateElements.Add(iMonth, monthPosition);
    iYear=CEikMfneNumber::NewL(aFont, aMinimumYear, aMaximumYear, initialDate.Year(), flags|CEikMfneNumber::ERepresentsYear);
    dateElements.Add(iYear, yearPosition);
    // add first date element
    iOwner.AddField(dateElements.Next());
    // add first separator
    HBufC* firstSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
    firstSeparatorText->Des().Append(locale.DateSeparator(dateSeparatorIndex++));
    iOwner.AddField(CEikMfneSeparator::NewL(firstSeparatorText));
    CleanupStack::Pop(firstSeparatorText);

    // add second date element
    iOwner.AddField(dateElements.Next());
    // add second separator
    HBufC* secondSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
    secondSeparatorText->Des().Append(locale.DateSeparator(dateSeparatorIndex++));
    iOwner.AddField(CEikMfneSeparator::NewL(secondSeparatorText));
    CleanupStack::Pop(secondSeparatorText); 

    // add third date element
    iOwner.AddField(dateElements.Next());
    CleanupStack::PopAndDestroy(); // 
    iOwner.SetValidateCallBack( TCallBack( ValidateCallBack, this ) );
    }

void CDateEditor::RefreshFromLocale(const CFont& aFont)
/**
Update the date editor from locale - ensursed that digits are displayed using correct numerals for current device settings.
@publishedComponent
@param aFont the font of the MFNE which owns this class
@post the next time the editor is drawn it will display the number using digit type derived from the locale information.
*/
    {
    iDay->RefreshDigitType(aFont);
    iMonth->RefreshDigitType(aFont);
    iYear->RefreshDigitType(aFont);
    }

void CDateEditor::SetMinimumAndMaximum(TInt aMinimumYear, TInt aMaximumYear, const CFont& aFont)
    {
    iYear->SetMinimumAndMaximum(aMinimumYear, aMaximumYear, aFont);
    }

void CDateEditor::SetDate(const TTime& aDate, const CFont& aFont)
    {
    TDateTime date=aDate.DateTime();
    iDay->SetValue(date.Day()+1, aFont);
    iMonth->SetValue(date.Month()+1, aFont);
    iYear->SetValue(date.Year(), aFont);
    }

TTime CDateEditor::Date(const TTime* aTime) const
    {
    TInt hour=0;
    TInt minute=0;
    TInt second=0;
    if (aTime)
        {
        TDateTime time=aTime->DateTime();
        hour=time.Hour();
        minute=time.Minute();
        second=time.Second();
        }
    TDateTime date;
    TInt day = iDay->Value();
    if (date.Set(iYear->Value(), (TMonth)(iMonth->Value()-1), day-1, 0, 0, 0, 0)!=KErrNone)
        {
        TInt maximumDay = Time::DaysInMonth( iYear->Value(), (TMonth)(iMonth->Value()-1));
        if (day > maximumDay)
            {
            day = maximumDay;
            iDay->SetValue(maximumDay, *(iOwner.Font()));
            iOwner.HighlightField(iDayFieldPosition + iFirstFieldPosition);
            }
//      AVKON - should not leave in Date() function. Day value is fixed instead.
        }
    return TTime(TDateTime(iYear->Value(), (TMonth)(iMonth->Value()-1), day-1, hour, minute, second, 0));
    }

TKeyResponse CDateEditor::OfferKeyEventL(const TKeyEvent& /*aKeyEvent*/, TEventCode /*aType*/)
    {
//  AVKON - we do not want any behavior from up/down or tab key (which was here before)
    return EKeyWasNotConsumed;
    }

void CDateEditor::PrepareForFocusLossL()
    {
    TDateTime date;

    if (date.Set(iYear->Value(), (TMonth)(iMonth->Value()-1), iDay->Value()-1, 0, 0, 0, 0)!=KErrNone)
        {
        TInt maximumDay = Time::DaysInMonth( iYear->Value(), (TMonth)(iMonth->Value()-1));
        if ( iDay->Value() > maximumDay )
            {
            iDay->SetValue(maximumDay, *(iOwner.Font()));    
            iOwner.HighlightField(iDayFieldPosition + iFirstFieldPosition);
            iOwner.ReportUpdate();
            }
         iOwner.DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_DATE_INVALID, date);
        //aEikonEnv.LeaveWithInfoMsg(R_EIK_TBUF_DATE_INVALID);
        }
    }

void CDateEditor::CreatePopoutIfRequiredL()
    {
    if (!iWithoutPopoutCalendar)
        ExecuteCalendarL();
    }

void CDateEditor::ExecuteCalendarL()
    {
    if( AknLayoutUtils::PenEnabled() )
        {
        // must not do anything when Touch UI is enabled
        }
    else
        {
        User::Leave(KErrNotSupported);
        }

/*  Build Thinning
    CEikCalendar* calendar=new(ELeave) CEikCalendar;
    calendar->SetCalendarObserver(iCalendarObserver);
    calendar->ExecuteLD(R_EIK_ONE_MONTH_CALENDAR);
*/
    }

TInt CDateEditor::Year() const
    {
    return(iYear->Value());
    }

TInt CDateEditor::Month() const
    {
    return(iMonth->Value());
    }

TInt CDateEditor::Day() const
    {
    return(iDay->Value());
    }

TBool CDateEditor::IsTextNull() const
    {
    return ( iYear->IsTextNull() | iMonth->IsTextNull() |  iDay->IsTextNull());
    }

void CDateEditor::UpdateMaxDay( const CFont& aFont )
	{
	TInt maximumDay = Time::DaysInMonth( iYear->Value(), (TMonth)(iMonth->Value()-1));
	iDay->SetMinimumAndMaximum( 1, maximumDay, aFont );
	}

// CDateEditor::RDateElements

CDateEditor::RDateElements::RDateElements()
    {
    for (TInt i=0; i<EMaximumNumberOfDateElements; ++i)
        iDateElements[i]=NULL;
    }

void CDateEditor::RDateElements::Add(CEikMfneNumber* aDateElement, TInt aPosition)
    {
    __ASSERT_DEBUG((aPosition>=0) && (aPosition<EMaximumNumberOfDateElements), Panic(EEikPanicDateEditorBadDateElementPosition));
    iDateElements[aPosition]=aDateElement;
    }

CEikMfneNumber* CDateEditor::RDateElements::Next()
    {
    for (TInt i=0; i<EMaximumNumberOfDateElements; ++i)
        if (iDateElements[i])
            {
            CEikMfneNumber* dateElement=iDateElements[i];
            iDateElements[i]=NULL;
            return dateElement;
            }
#if defined(_DEBUG)
    Panic(EEikPanicDateEditorNoMoreDateElements);
#endif
    return NULL; // dummy return to prevent compiler error
    }

void CDateEditor::RDateElements::Cleanup(TAny* aThis)
    {
    for (TInt i=0; i<EMaximumNumberOfDateElements; ++i)
        delete ((RDateElements*)aThis)->iDateElements[i];
    }

void CDateEditor::SetUninitialised(TBool aUninitialised)
    {
    if (iDay)
        iDay->SetUninitialised(aUninitialised);
    if (iMonth)
        iMonth->SetUninitialised(aUninitialised);
    if (iYear)
        iYear->SetUninitialised(aUninitialised);
    }

TBool CDateEditor::IsUninitialised() const
    {
    return 
        (iDay && iDay->IsUninitialised()) ||
        (iMonth && iMonth->IsUninitialised()) ||
        (iYear && iYear->IsUninitialised());
    }

TInt CDateEditor::ValidateCallBack( TAny* aPtr )
    {
    CDateEditor* edit( static_cast<CDateEditor*>( aPtr ) );
    TRAPD( err, edit->PrepareForFocusLossL() );
    edit->UpdateMaxDay( *(CEikonEnv::Static()->NormalFont()));
    return err;
    }

// CEikTimeEditor

EXPORT_C CEikTimeEditor::CEikTimeEditor()
    {
    __DECLARE_NAME(_S("CEikTimeEditor"));
    AKNTASHOOK_ADD( this, "CEikTimeEditor" );
    }

EXPORT_C CEikTimeEditor::~CEikTimeEditor()
    {
    AKNTASHOOK_REMOVE();
    delete iTimeEditor;
    }

EXPORT_C void CEikTimeEditor::ConstructL(const TTime& aMinimumTime, const TTime& aMaximumTime, const TTime& aInitialTime, TUint32 aFlags)
    {
    iTimeEditor=new(ELeave) CTimeEditor;
    //CreateFieldArrayL(iTimeEditor->NumFieldsRequired(aWithoutSecondsField? CTimeEditor::EWithoutSecondsField: 0));
    CreateFieldArrayL(iTimeEditor->NumFieldsRequired(aFlags));
    iTimeEditor->ConstructFieldsL(*this, aInitialTime, *Font());
    iTimeEditor->RefreshFromLocale(*Font());
    DoSetMinimumAndMaximum(aMinimumTime, aMaximumTime);
    }

EXPORT_C void CEikTimeEditor::SetMinimumAndMaximum(const TTime& aMinimumTime, const TTime& aMaximumTime)
    {
    // __ASSERT_ALWAYS((aMinimumTime>=iMinimumTime) && (aMaximumTime<=iMaximumTime), Panic(EEikPanicTimeEditorBadMinimumAndMaximum1));
    DoSetMinimumAndMaximum(aMinimumTime, aMaximumTime);
    }

EXPORT_C void CEikTimeEditor::GetMinimumAndMaximum(TTime& aMinimumTime, TTime& aMaximumTime) const
    {
    aMinimumTime=iMinimumTime;
    aMaximumTime=iMaximumTime;
    }

EXPORT_C void CEikTimeEditor::SetTime(const TTime& aTime)
    {
    iTimeEditor->SetTime(aTime, *Font());
	TRAP_IGNORE ( ReportEventL( MCoeControlObserver::EEventStateChanged ) );
    }

EXPORT_C TTime CEikTimeEditor::Time() const
    {
    return iTimeEditor->Time();
    }

void CEikTimeEditor::SetTTime(const TTime& aTime)
    {
    SetTime(aTime);
    }

TTime CEikTimeEditor::GetTTime() const
    {
    return(Time());
    }

EXPORT_C void CEikTimeEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TTime minimumTime=ReadTime(aResourceReader);
    TTime maximumTime=ReadTime(aResourceReader);
    TUint32 flags=aResourceReader.ReadUint8();
    ConstructL(minimumTime, maximumTime, maximumTime, flags);
    }

EXPORT_C void CEikTimeEditor::PrepareForFocusLossL()
    {
    CEikMfne::PrepareForFocusLossL();
    TTime time=Time();
    if (time<iMinimumTime)
        {
        SetTime(iMinimumTime);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_BEFORE_EARLIEST_ALLOWED_TIME, iMinimumTime);
        }
    else if (time>iMaximumTime)
        {
        SetTime(iMaximumTime);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_AFTER_LATEST_ALLOWED_TIME, iMaximumTime);
        }
    }

void CEikTimeEditor::DoSetMinimumAndMaximum(const TTime& aMinimumTime, const TTime& aMaximumTime)
    {
    __ASSERT_ALWAYS(aMinimumTime<=aMaximumTime, Panic(EEikPanicTimeEditorBadMinimumAndMaximum2));
    iMinimumTime=aMinimumTime;
    iMaximumTime=aMaximumTime;
    TTime time=Time();
    if (time<iMinimumTime)
        SetTime(iMinimumTime);
    else if (time>iMaximumTime)
        SetTime(iMaximumTime);
    }

EXPORT_C void CEikTimeEditor::CEikMfne_Reserved()
    {
    }
    
EXPORT_C void CEikTimeEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikTTimeEditor::HandlePointerEventL(aPointerEvent); 
    }   

EXPORT_C void* CEikTimeEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikTimeEditor::SetUninitialised(TBool aUninitialised)
    {
    iTimeEditor->SetUninitialised(aUninitialised);
    }

EXPORT_C TBool CEikTimeEditor::IsUninitialised() const
    {
    return iTimeEditor->IsUninitialised();
    }

// CEikDateEditor

EXPORT_C CEikDateEditor::CEikDateEditor()
    {
    __DECLARE_NAME(_S("CEikDateEditor"));
    AKNTASHOOK_ADD( this, "CEikDateEditor" );
    }

EXPORT_C CEikDateEditor::~CEikDateEditor()
    {
    AKNTASHOOK_REMOVE();
    delete iDateEditor;
    }

EXPORT_C void CEikDateEditor::ConstructL(const TTime& aMinimumDate, const TTime& aMaximumDate, const TTime& aInitialDate, TBool aWithoutPopoutCalendar)
    {
    __ASSERT_DEBUG(KAknMinimumDate<=aMinimumDate, Panic(EEikPanicDateEditorBadMinimum));
    __ASSERT_DEBUG(aMaximumDate<=KAknMaximumDate, Panic(EEikPanicDateEditorBadMaximum));
    TTime minDate = Max(KAknMinimumDate, aMinimumDate);
    TTime maxDate = Min(aMaximumDate, KAknMaximumDate);
    iDateEditor=new(ELeave) CDateEditor(*this,this,0);
    CreateFieldArrayL(iDateEditor->NumFieldsRequired());
    iDateEditor->ConstructFieldsL(minDate.DateTime().Year(), maxDate.DateTime().Year(), aInitialDate, aWithoutPopoutCalendar, *iEikonEnv, *Font());
    DoSetMinimumAndMaximum(minDate, maxDate);
    iDateEditor->RefreshFromLocale(*Font());
    }

EXPORT_C void CEikDateEditor::SetMinimumAndMaximum(const TTime& aMinimumDate, const TTime& aMaximumDate)
    {
    __ASSERT_DEBUG(KAknMinimumDate<=aMinimumDate, Panic(EEikPanicDateEditorBadMinimum));
    __ASSERT_DEBUG(aMaximumDate<=KAknMaximumDate, Panic(EEikPanicDateEditorBadMaximum));
    TTime minDate = Max(KAknMinimumDate, aMinimumDate);
    TTime maxDate = Min(aMaximumDate, KAknMaximumDate);
    iDateEditor->SetMinimumAndMaximum(minDate.DateTime().Year(), maxDate.DateTime().Year(), *Font());
    DoSetMinimumAndMaximum(minDate, maxDate);
    }

EXPORT_C void CEikDateEditor::GetMinimumAndMaximum(TTime& aMinimumDate, TTime& aMaximumDate) const
    {
    aMinimumDate=iMinimumDate;
    aMaximumDate=iMaximumDate;
    }

EXPORT_C void CEikDateEditor::SetDate(const TTime& aDate)
    {
    iDateEditor->SetDate(aDate, *Font());
    TRAP_IGNORE ( ReportEventL( MCoeControlObserver::EEventStateChanged ) );
    }

EXPORT_C TTime CEikDateEditor::Date() const
    {
    return iDateEditor->Date();
    }

void CEikDateEditor::SetTTime(const TTime& aTime)
    {
    SetDate(aTime);
    }

TTime CEikDateEditor::GetTTime() const
    {
    return(Date());
    }

EXPORT_C TKeyResponse CEikDateEditor::OfferKeyEventL( const TKeyEvent& aKeyEvent, TEventCode aType )
    {
	TKeyResponse ret = EKeyWasNotConsumed;
    if ( iDateEditor->OfferKeyEventL( aKeyEvent, aType ) == EKeyWasConsumed )
    	{
        ret = EKeyWasConsumed;
    	}
    else 
    	{
	    ret = CEikMfne::OfferKeyEventL( aKeyEvent, aType );
	    if ( ( aType == EEventKey ) && 
		     ( aKeyEvent.iCode == EKeyDownArrow || aKeyEvent.iCode == EKeyUpArrow ) )
		    {
		    iDateEditor->UpdateMaxDay( *Font() );
		    this->DrawDeferred();
		    }
    	}
    return ret;
    }

EXPORT_C void CEikDateEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TTime minimumDate=ReadDate(aResourceReader);
    TTime maximumDate=ReadDate(aResourceReader);
    TUint32 flags=aResourceReader.ReadUint8();
    ConstructL(minimumDate, maximumDate, maximumDate, flags&EEikDateWithoutPopoutCalendar);
    }

EXPORT_C void CEikDateEditor::PrepareForFocusLossL()
    {
    CEikMfne::PrepareForFocusLossL();
    iDateEditor->PrepareForFocusLossL();
    TTime date=Date();
    if (date<iMinimumDate)
        {
        SetDate(iMinimumDate);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_BEFORE_EARLIEST_ALLOWED_DATE, iMinimumDate);
        }
    else if (date>iMaximumDate)
        {
        SetDate(iMaximumDate);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_AFTER_LATEST_ALLOWED_DATE, iMaximumDate);
        }
    }

void CEikDateEditor::CreatePopoutIfRequiredL()
    {
    iDateEditor->CreatePopoutIfRequiredL();
    }

void CEikDateEditor::GetMinimumAndMaximumAndInitialDatesForCalendarL(TTime& aMinimumDate, TTime& aMaximumDate, TTime& aInitialDate) const
    {
    aMinimumDate=iMinimumDate;
    aMaximumDate=iMaximumDate;
    TDateTime date;
    if (iDateEditor->IsTextNull())
        CEikMfne::LeaveWithAlert(R_EIK_TBUF_DATE_INVALID);
    if (date.Set(iDateEditor->Year(), (TMonth)(iDateEditor->Month()-1), iDateEditor->Day()-1, 0, 0, 0, 0)!=KErrNone)
        CEikMfne::LeaveWithAlert(R_EIK_TBUF_DATE_INVALID);
    aInitialDate=iDateEditor->Date();
    }

void CEikDateEditor::SetDateFromCalendarAndDrawNow(const TTime& aDate)
    {
    iDateEditor->SetDate(aDate, *Font());
    DrawNow();
    }

void CEikDateEditor::DoSetMinimumAndMaximum(const TTime& aMinimumDate, const TTime& aMaximumDate)
    {
    __ASSERT_ALWAYS(aMinimumDate<=aMaximumDate, Panic(EEikPanicDateEditorBadMinimumAndMaximum2));
    TDateTime minimumDate=aMinimumDate.DateTime();
    TDateTime maximumDate=aMaximumDate.DateTime();
    iMinimumDate=TDateTime(minimumDate.Year(), minimumDate.Month(), minimumDate.Day(), 0, 0, 0, 0);
    iMaximumDate=TDateTime(maximumDate.Year(), maximumDate.Month(), maximumDate.Day(), 0, 0, 0, 0);
    TTime date=Date();
    if (date<iMinimumDate)
        SetDate(iMinimumDate);
    else if (date>iMaximumDate)
        SetDate(iMaximumDate);
    }

EXPORT_C void CEikDateEditor::CEikMfne_Reserved()
    {
    }
    
EXPORT_C void CEikDateEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikTTimeEditor::HandlePointerEventL(aPointerEvent); 
    }   

EXPORT_C void* CEikDateEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikDateEditor::SetUninitialised(TBool aUninitialised)
    {
    iDateEditor->SetUninitialised(aUninitialised);
    }

EXPORT_C TBool CEikDateEditor::IsUninitialised() const
    {
    return iDateEditor->IsUninitialised();
    }


// CEikTimeAndDateEditor

EXPORT_C CEikTimeAndDateEditor::CEikTimeAndDateEditor()
    {
    __DECLARE_NAME(_S("CEikTimeAndDateEditor"));
    AKNTASHOOK_ADD( this, "CEikTimeAndDateEditor" );
    }

EXPORT_C CEikTimeAndDateEditor::~CEikTimeAndDateEditor()
    {
    AKNTASHOOK_REMOVE();
    delete iTimeEditor;
    delete iDateEditor;
    }

EXPORT_C void CEikTimeAndDateEditor::ConstructL(const TTime& aMinimumTimeAndDate, const TTime& aMaximumTimeAndDate,
                                    const TTime& aInitialTimeAndDate,TUint32 aFlags, HBufC* aInterveningText)
    {
    iTimeEditor=new(ELeave) CTimeEditor;
    TInt numTimeFields(iTimeEditor->NumFieldsRequired(aFlags));
    iDateEditor=new(ELeave) CDateEditor(*this,this,numTimeFields);
    CreateFieldArrayL(numTimeFields+(aInterveningText? 1: 0)+iDateEditor->NumFieldsRequired());
    const CFont& font=*Font();
    CEikMfneSeparator* interveningText=NULL;

    // Implement reversed time and date date for A&H locales
    if ( TBidiText::ScriptDirectionality( (TLanguage)(User::Language() & KAknLanguageMask) ) == TBidiText::ELeftToRight)
        {
        iTimeEditor->ConstructFieldsL(*this, aInitialTimeAndDate, font);
        iTimeEditor->RefreshFromLocale(font);

        if (aInterveningText)
            {
            interveningText=CEikMfneSeparator::NewL(NULL);
            AddField(interveningText);
            }
        iDateEditor->ConstructFieldsL(aMinimumTimeAndDate.DateTime().Year(), aMaximumTimeAndDate.DateTime().Year(), aInitialTimeAndDate, (aFlags&CTimeEditor::EWithoutPopoutCalendar), *iEikonEnv, font);
        iDateEditor->RefreshFromLocale(font);
        }
    else  // These are reversed
        {
        iDateEditor->ConstructFieldsL(aMinimumTimeAndDate.DateTime().Year(), aMaximumTimeAndDate.DateTime().Year(), aInitialTimeAndDate, (aFlags&CTimeEditor::EWithoutPopoutCalendar), *iEikonEnv, font);
        iDateEditor->RefreshFromLocale(font);

        if (aInterveningText)
            {
            interveningText=CEikMfneSeparator::NewL(NULL);
            AddField(interveningText);
            }

        iTimeEditor->ConstructFieldsL(*this, aInitialTimeAndDate, font);
        iTimeEditor->RefreshFromLocale(font);
        }

    DoSetMinimumAndMaximum(aMinimumTimeAndDate, aMaximumTimeAndDate);
    // do stuff that can only be done when all leaving functions have successfully been done
    if (aInterveningText)
        interveningText->SetText(aInterveningText);
    }

EXPORT_C void CEikTimeAndDateEditor::SetMinimumAndMaximum(const TTime& aMinimumTimeAndDate, const TTime& aMaximumTimeAndDate)
    {
    // __ASSERT_ALWAYS((aMinimumTimeAndDate>=iMinimumTimeAndDate) && (aMaximumTimeAndDate<=iMaximumTimeAndDate), Panic(EEikPanicTimeAndDateEditorBadMinimumAndMaximum1));
    iDateEditor->SetMinimumAndMaximum(aMinimumTimeAndDate.DateTime().Year(), aMaximumTimeAndDate.DateTime().Year(), *Font());
    DoSetMinimumAndMaximum(aMinimumTimeAndDate, aMaximumTimeAndDate);
    }

EXPORT_C void CEikTimeAndDateEditor::GetMinimumAndMaximum(TTime& aMinimumTimeAndDate, TTime& aMaximumTimeAndDate) const
    {
    aMinimumTimeAndDate=iMinimumTimeAndDate;
    aMaximumTimeAndDate=iMaximumTimeAndDate;
    }

EXPORT_C void CEikTimeAndDateEditor::SetTimeAndDate(const TTime& aTimeAndDate)
    {
    const CFont& font=*Font();
    iDateEditor->SetDate(aTimeAndDate, font);
    TDateTime dateTime=aTimeAndDate.DateTime();
    dateTime.SetYear(0);
    dateTime.SetMonth(EJanuary);
    dateTime.SetDay(0);
    TTime timeWithoutDate(dateTime);
    iTimeEditor->SetTime(timeWithoutDate, font);
    }

EXPORT_C TTime CEikTimeAndDateEditor::TimeAndDate() const
    {
    TTime time=iTimeEditor->Time();
    return iDateEditor->Date(&time);
    }

EXPORT_C void CEikTimeAndDateEditor::SetUninitialised(TBool aUninitialised)
    {
    iTimeEditor->SetUninitialised(aUninitialised);
    iDateEditor->SetUninitialised(aUninitialised);
    }

EXPORT_C TBool CEikTimeAndDateEditor::IsUninitialised() const
    {
    return iTimeEditor->IsUninitialised() || iDateEditor->IsUninitialised();
    }


void CEikTimeAndDateEditor::SetTTime(const TTime& aTime)
    {
    SetTimeAndDate(aTime);
    }

TTime CEikTimeAndDateEditor::GetTTime() const
    {
    return(TimeAndDate());
    }

EXPORT_C TKeyResponse CEikTimeAndDateEditor::OfferKeyEventL(const TKeyEvent& aKeyEvent, TEventCode aType)
    {
    if (iDateEditor->OfferKeyEventL(aKeyEvent, aType)==EKeyWasConsumed)
        return EKeyWasConsumed;
    return CEikMfne::OfferKeyEventL(aKeyEvent, aType);
    }

EXPORT_C void CEikTimeAndDateEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TTime minimumTimeAndDate=ReadTimeAndDate(aResourceReader);
    TTime maximumTimeAndDate=ReadTimeAndDate(aResourceReader);
    TBool flags=aResourceReader.ReadUint8();
    HBufC* interveningText=aResourceReader.ReadHBufCL();
    CleanupStack::PushL(interveningText);
    ConstructL(minimumTimeAndDate, maximumTimeAndDate, maximumTimeAndDate, flags, interveningText);
    CleanupStack::Pop();
    }

EXPORT_C void CEikTimeAndDateEditor::PrepareForFocusLossL()
    {
    CEikMfne::PrepareForFocusLossL();
    iDateEditor->PrepareForFocusLossL();
    TTime timeAndDate=TimeAndDate();
    if (timeAndDate<iMinimumTimeAndDate)
        {
        SetTimeAndDate(iMinimumTimeAndDate);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_BEFORE_EARLIEST_ALLOWED_TIME_AND_DATE, iMinimumTimeAndDate);
        }
    else if (timeAndDate>iMaximumTimeAndDate)
        {
        SetTimeAndDate(iMaximumTimeAndDate);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_AFTER_LATEST_ALLOWED_TIME_AND_DATE, iMaximumTimeAndDate);
        }
    }

void CEikTimeAndDateEditor::CreatePopoutIfRequiredL()
    {
    iDateEditor->CreatePopoutIfRequiredL();
    }

void CEikTimeAndDateEditor::GetMinimumAndMaximumAndInitialDatesForCalendarL(TTime& aMinimumDate, TTime& aMaximumDate, TTime& aInitialDate) const
    {
    aMinimumDate=iMinimumTimeAndDate;
    aMaximumDate=iMaximumTimeAndDate;
    TDateTime date;
    if (iDateEditor->IsTextNull())
        CEikMfne::LeaveWithAlert(R_EIK_TBUF_DATE_INVALID);
    if (date.Set(iDateEditor->Year(), (TMonth)(iDateEditor->Month()-1), iDateEditor->Day()-1, 0, 0, 0, 0)!=KErrNone)
        CEikMfne::LeaveWithAlert(R_EIK_TBUF_DATE_INVALID);
    aInitialDate=iDateEditor->Date();
    }

void CEikTimeAndDateEditor::SetDateFromCalendarAndDrawNow(const TTime& aDate)
    {
    iDateEditor->SetDate(aDate, *Font());
    DrawNow();
    }

void CEikTimeAndDateEditor::DoSetMinimumAndMaximum(const TTime& aMinimumTimeAndDate, const TTime& aMaximumTimeAndDate)
    {
    __ASSERT_ALWAYS(aMinimumTimeAndDate<=aMaximumTimeAndDate, Panic(EEikPanicTimeAndDateEditorBadMinimumAndMaximum2));
    iMinimumTimeAndDate=aMinimumTimeAndDate;
    iMaximumTimeAndDate=aMaximumTimeAndDate;
    TTime timeAndDate=TimeAndDate();
    if (timeAndDate<iMinimumTimeAndDate)
        SetTimeAndDate(iMinimumTimeAndDate);
    else if (timeAndDate>iMaximumTimeAndDate)
        SetTimeAndDate(iMaximumTimeAndDate);
    }

EXPORT_C void CEikTimeAndDateEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikTTimeEditor::HandlePointerEventL(aPointerEvent); 
    }

EXPORT_C void* CEikTimeAndDateEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikTimeAndDateEditor::CEikMfne_Reserved()
    {
    }


// CEikDurationEditor

EXPORT_C CEikDurationEditor::CEikDurationEditor()
    {
    __DECLARE_NAME(_S("CEikDurationEditor"));
    AKNTASHOOK_ADD( this, "CEikDurationEditor" );
    }

EXPORT_C CEikDurationEditor::~CEikDurationEditor()
    {
    AKNTASHOOK_REMOVE();
    delete iTimeEditor;
    }

EXPORT_C void CEikDurationEditor::ConstructL(const TTimeIntervalSeconds& aMinimumDuration, const TTimeIntervalSeconds& aMaximumDuration, const TTimeIntervalSeconds& aInitialDuration, TUint32 aFlags)
    {
    iTimeEditor=new(ELeave) CTimeEditor;
    aFlags|=CTimeEditor::EForce24HourFormat;
    CreateFieldArrayL(iTimeEditor->NumFieldsRequired(aFlags));
    iTimeEditor->ConstructFieldsL(*this, Convert(aInitialDuration), *Font());
    iTimeEditor->RefreshFromLocale(*Font());

    DoSetMinimumAndMaximum(aMinimumDuration, aMaximumDuration);
    }

EXPORT_C void CEikDurationEditor::SetMinimumAndMaximum(const TTimeIntervalSeconds& aMinimumDuration, const TTimeIntervalSeconds& aMaximumDuration)
    {
    // __ASSERT_ALWAYS((aMinimumDuration>=iMinimumDuration) && (aMaximumDuration<=iMaximumDuration), Panic(EEikPanicDurationEditorBadMinimumAndMaximum1));
    DoSetMinimumAndMaximum(aMinimumDuration, aMaximumDuration);
    }

EXPORT_C void CEikDurationEditor::GetMinimumAndMaximum(TTimeIntervalSeconds& aMinimumDuration, TTimeIntervalSeconds& aMaximumDuration) const
    {
    aMinimumDuration=iMinimumDuration;
    aMaximumDuration=iMaximumDuration;
    }

EXPORT_C void CEikDurationEditor::SetDuration(const TTimeIntervalSeconds& aDuration)
    {
    iTimeEditor->SetTime(Convert(aDuration), *Font());
    }

EXPORT_C TTimeIntervalSeconds CEikDurationEditor::Duration() const
    {
    return Convert(iTimeEditor->Time());
    }

EXPORT_C void CEikDurationEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TTimeIntervalSeconds minimumDuration=ReadDuration(aResourceReader);
    TTimeIntervalSeconds maximumDuration=ReadDuration(aResourceReader);
    TUint32 flags=aResourceReader.ReadUint8();
    ConstructL(minimumDuration, maximumDuration, maximumDuration, flags);
    }

EXPORT_C void CEikDurationEditor::PrepareForFocusLossL()
    {
    CEikMfne::PrepareForFocusLossL();
    TTimeIntervalSeconds duration=Duration();
    if (duration<iMinimumDuration)
        {
        SetDuration(iMinimumDuration);

        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_BELOW_SHORTEST_ALLOWED_DURATION, Convert(iMinimumDuration));
        }
    else if (duration>iMaximumDuration)
        {
        SetDuration(iMaximumDuration);
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(R_EIK_TBUF_ABOVE_LONGEST_ALLOWED_DURATION, Convert(iMaximumDuration));
        }
    }

void CEikDurationEditor::DoSetMinimumAndMaximum(const TTimeIntervalSeconds& aMinimumDuration, const TTimeIntervalSeconds& aMaximumDuration)
    {
    __ASSERT_ALWAYS(aMinimumDuration<=aMaximumDuration, Panic(EEikPanicDurationEditorBadMinimumAndMaximum2));
    iMinimumDuration=aMinimumDuration;
    iMaximumDuration=aMaximumDuration;
    TTimeIntervalSeconds duration=Duration();
    if (duration<iMinimumDuration)
        SetDuration(iMinimumDuration);
    else if (duration>iMaximumDuration)
        SetDuration(iMaximumDuration);
    }
    
EXPORT_C void CEikDurationEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikMfne::HandlePointerEventL(aPointerEvent); 
    }    

EXPORT_C void* CEikDurationEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikDurationEditor::CEikMfne_Reserved()
    {
    }


// CEikTimeOffsetEditor

EXPORT_C CEikTimeOffsetEditor::CEikTimeOffsetEditor()
    {
    __DECLARE_NAME(_S("CEikTimeOffsetEditor"));
    AKNTASHOOK_ADD( this, "CEikTimeOffsetEditor" );
    }

EXPORT_C CEikTimeOffsetEditor::~CEikTimeOffsetEditor()
    {
    AKNTASHOOK_REMOVE();
    delete iTimeEditor;
    }

EXPORT_C void CEikTimeOffsetEditor::ConstructL(const TTimeIntervalSeconds& aMinimumTimeOffset, const TTimeIntervalSeconds& aMaximumTimeOffset, const TTimeIntervalSeconds& aInitialTimeOffset, TUint32 aFlags)
    {
    TTimeIntervalSeconds magnitudeOfInitialTimeOffset=aInitialTimeOffset;
    TBool positive=ETrue;
    if (magnitudeOfInitialTimeOffset.Int()<0)
        {
        magnitudeOfInitialTimeOffset=-magnitudeOfInitialTimeOffset.Int();
        positive=EFalse;
        }
    iTimeEditor=new(ELeave) CTimeEditor;
    if (~aFlags&CTimeEditor::ETimeZoneOffsetFormat)
        {
        aFlags|=CTimeEditor::EForce24HourFormat;
        }
    CreateFieldArrayL(2+iTimeEditor->NumFieldsRequired(aFlags));
    iSign=CEikMfneSymbol::NewL(2);
    CleanupStack::PushL(iSign); // this needs to be pushed as it would not be destroyed by the class' destructor
    // add the "+" symbol
    HBufC* plusText=HBufC::NewLC(1); // also pushes on to CleanupStack
    *plusText=KPlus;
    iSign->AddSymbolicItem(CEikMfneSymbol::CItem::NewL('+', '+', plusText), positive);
    CleanupStack::Pop();
    // add the "-" symbol
    HBufC* minusText=HBufC::NewLC(1); // also pushes on to CleanupStack
    *minusText=KMinus;
    iSign->AddSymbolicItem(CEikMfneSymbol::CItem::NewL('-', '-', minusText), !positive);
    CleanupStack::Pop(2);
    AddField(iSign);
    HBufC* spaceSeparatorText=HBufC::NewLC(1); // also pushes on to CleanupStack
    spaceSeparatorText->Des().Append(TChar(' '));
    AddField(CEikMfneSeparator::NewL(spaceSeparatorText));
    CleanupStack::Pop();
    iTimeEditor->ConstructFieldsL(*this, Convert(magnitudeOfInitialTimeOffset), *Font());
    DoSetMinimumAndMaximum(aMinimumTimeOffset, aMaximumTimeOffset);
    }

EXPORT_C void CEikTimeOffsetEditor::SetMinimumAndMaximum(const TTimeIntervalSeconds& aMinimumTimeOffset, const TTimeIntervalSeconds& aMaximumTimeOffset)
    {
    // __ASSERT_ALWAYS((aMinimumTimeOffset>=iMinimumTimeOffset) && (aMaximumTimeOffset<=iMaximumTimeOffset), Panic(EEikPanicTimeOffsetEditorBadMinimumAndMaximum1));
    DoSetMinimumAndMaximum(aMinimumTimeOffset, aMaximumTimeOffset);
    }

EXPORT_C void CEikTimeOffsetEditor::GetMinimumAndMaximum(TTimeIntervalSeconds& aMinimumTimeOffset, TTimeIntervalSeconds& aMaximumTimeOffset) const
    {
    aMinimumTimeOffset=iMinimumTimeOffset;
    aMaximumTimeOffset=iMaximumTimeOffset;
    }

EXPORT_C void CEikTimeOffsetEditor::SetTimeOffset(const TTimeIntervalSeconds& aTimeOffset)
    {
    TTimeIntervalSeconds magnitudeOfTimeOffset=aTimeOffset;
    TInt signId='+';
    if (magnitudeOfTimeOffset.Int()<0)
        {
        magnitudeOfTimeOffset=-magnitudeOfTimeOffset.Int();
        signId='-';
        }
    iTimeEditor->SetTime(Convert(magnitudeOfTimeOffset), *Font());
    iSign->SetCurrentSymbolicItemToId(signId);
    }

EXPORT_C TTimeIntervalSeconds CEikTimeOffsetEditor::TimeOffset() const
    {
    TTimeIntervalSeconds timeOffset=Convert(iTimeEditor->Time());
    if (iSign->IdOfCurrentSymbolicItem()=='-')
        timeOffset=-timeOffset.Int();
    return timeOffset;
    }

EXPORT_C void CEikTimeOffsetEditor::ConstructFromResourceL(TResourceReader& aResourceReader)
    {
    TTimeIntervalSeconds minimumTimeOffset=ReadTimeOffset(aResourceReader);
    TTimeIntervalSeconds maximumTimeOffset=ReadTimeOffset(aResourceReader);
    TUint32 flags=aResourceReader.ReadUint8();
    ConstructL(minimumTimeOffset, maximumTimeOffset, maximumTimeOffset, flags);
    }

EXPORT_C void CEikTimeOffsetEditor::PrepareForFocusLossL()
    {
    CEikMfne::PrepareForFocusLossL();
    TTimeIntervalSeconds timeOffset=TimeOffset();
    if (timeOffset<iMinimumTimeOffset)
        {
        SetTimeOffset(iMinimumTimeOffset);
        TInt valueOfMinimumTimeOffset=iMinimumTimeOffset.Int();
        TInt resourceId=R_EIK_TBUF_BELOW_MINIMUM_TIME_OFFSET;
        if (valueOfMinimumTimeOffset<0)
            {
            valueOfMinimumTimeOffset=-valueOfMinimumTimeOffset;
            resourceId=R_EIK_TBUF_BELOW_NEGATIVE_MINIMUM_TIME_OFFSET;
            }
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(resourceId, Convert(TTimeIntervalSeconds(valueOfMinimumTimeOffset)));
        }
    else if (timeOffset>iMaximumTimeOffset)
        {
        SetTimeOffset(iMaximumTimeOffset);
        TInt valueOfMaximumTimeOffset=iMaximumTimeOffset.Int();
        TInt resourceId=R_EIK_TBUF_ABOVE_MAXIMUM_TIME_OFFSET;
        if (valueOfMaximumTimeOffset<0)
            {
            valueOfMaximumTimeOffset=-valueOfMaximumTimeOffset;
            resourceId=R_EIK_TBUF_ABOVE_NEGATIVE_MAXIMUM_TIME_OFFSET;
            }
        DrawNowAndLeaveWithTimeDateFormatInfoMsgL(resourceId, Convert(TTimeIntervalSeconds(valueOfMaximumTimeOffset)));
        }
    }

void CEikTimeOffsetEditor::DoSetMinimumAndMaximum(const TTimeIntervalSeconds& aMinimumTimeOffset, const TTimeIntervalSeconds& aMaximumTimeOffset)
    {
    __ASSERT_ALWAYS(aMinimumTimeOffset<=aMaximumTimeOffset, Panic(EEikPanicTimeOffsetEditorBadMinimumAndMaximum2));
    iMinimumTimeOffset=aMinimumTimeOffset;
    iMaximumTimeOffset=aMaximumTimeOffset;
    TTimeIntervalSeconds timeOffset=TimeOffset();
    if (timeOffset<iMinimumTimeOffset)
        SetTimeOffset(iMinimumTimeOffset);
    else if (timeOffset>iMaximumTimeOffset)
        SetTimeOffset(iMaximumTimeOffset);
    }

EXPORT_C void CEikTimeOffsetEditor::HandlePointerEventL(const TPointerEvent& aPointerEvent) 
    { 
    CEikMfne::HandlePointerEventL(aPointerEvent); 
    }

EXPORT_C void* CEikTimeOffsetEditor::ExtensionInterface( TUid /*aInterface*/ )
    {
    return NULL;
    }

EXPORT_C void CEikTimeOffsetEditor::CEikMfne_Reserved()
    {
    }

EXPORT_C void CEikMfne::SetSkinTextColorL(TInt aAknSkinIDForTextColor, TInt aAknSkinIDForBgColor)
    {
    CreateExtensionIfRequiredL();
        
    iExtension->iAknSkinColorIndex = aAknSkinIDForTextColor;
    iExtension->iSkinIdForTextBackgroundColor = aAknSkinIDForBgColor;
    
    // if we are already visible, try to apply new color
    if (IsVisible())
        {
        DrawDeferred(); // No hurry anyway
        }
    }
    
EXPORT_C void CEikMfne::SizeChanged()
    {    
    CEikBorderedControl::SizeChanged();
    if ( !iFields || iCurrentField < 0 || iCurrentField >= iNumFields )
        {
        return;
        }
    TBool cursorDisplay( iFields[iCurrentField]->HighlightType() ==
        CEikMfneField::ECursor );
    if ( IsFocused() && cursorDisplay )
        {
        TRect mainPaneRect;        
        AknLayoutUtils::LayoutMetricsRect( AknLayoutUtils::EMainPane, 
            mainPaneRect );
        TRect rect( Rect() );
        if ( rect.iTl.iY < 0 || rect.iBr.iY > mainPaneRect.Height() ||
             rect.iTl.iX >= mainPaneRect.Width() || rect.iBr.iX <= 0 )
            {
            HideCursor();    
            }
        else
            {
            DrawCursor();
            }
        }    
    }

EXPORT_C void CEikMfne::MakeVisible( TBool aVisible )
    {
    CEikBorderedControl::MakeVisible( aVisible );
    if ( !aVisible && iExtension && iExtension->iCursorShown )
        {
        HideCursor();
        }
    
    if ( iExtension && !aVisible )
        {
        TRAP_IGNORE (
            iExtension->iExtendedInputCapabilities->ReportEventL( 
                CAknExtendedInputCapabilities::MAknEventObserver::EClosePenInputRequest, 
                0 );
            )
        }
    }

void CEikMfne::SetValidateCallBack( TCallBack aCallBack )
    {
    if ( iExtension )
        {
        iExtension->iValidateValueCallBack = aCallBack;
        }
    }

void CEikMfne::ReportStateChangeEventL()
    {
    ReportEventL( MCoeControlObserver::EEventStateChanged );
    }

void CEikMfne::LaunchPenInputL()
    {
    if ( iExtension && !iExtension->iDisablePenInput )
        {
        if ( iExtension->iFingerSupport && 
            iExtension->iFingerParam == EnableWithAllHighlight )
            {            
            iExtension->iHighlightAll = !iExtension->iTouchActivated &&
            !(iExtension->iExtendedInputCapabilities->Capabilities() & CAknExtendedInputCapabilities::EInputEditorQwertyInputActive);
            if ( iExtension->iHighlightAll )
                {                
                SetFirstEditableField();
                }            
            ReportUpdate();
            DrawDeferred();
            }
        iExtension->iExtendedInputCapabilities->ReportEventL( 
            CAknExtendedInputCapabilities::MAknEventObserver::EActivatePenInputRequest, 
            0 );
        iExtension->iTouchActivated = ETrue;
        }
    }