fep/aknfep/src/AknFepFnKeyManager.cpp
changeset 0 eb1f2e154e89
child 10 6defe5d1bd39
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/fep/aknfep/src/AknFepFnKeyManager.cpp	Tue Feb 02 01:02:04 2010 +0200
@@ -0,0 +1,620 @@
+/*
+* Copyright (c) 2002-2007 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:            Provides CAknFepManager definition
+*
+*/
+
+
+
+
+
+
+
+
+
+
+
+
+#ifdef _DEBUG
+#include <eikenv.h>
+#endif
+
+#include <fepbase.h>
+#include <avkon.hrh>
+#include <aknedsts.h>               //CAknEdwinState
+
+#include "AknFepFnKeyManager.h"
+#include "AknFepManager.h"
+#include <centralrepository.h>
+#include <AknFepInternalCRKeys.h>
+#include "PtiEngine.h"
+
+
+/*this part is removed 
+const   TUid    KPhoneUidAppPhone = { 0x100058B3 };
+const   TUid    KUidCalc = { 0x10005902 };
+const   TUid    KUidIdle = { 0x101fd64c };
+*/
+const   TStdScanCode stdFnKeyCode(EStdKeyRightFunc);    // Defines the std key code for Fn key
+
+////////////////////////////////////////////////////////////
+//   CAknFepFnKeyManager implementation
+//////////////////////////////////////////////////////////// 
+CAknFepFnKeyManager::CAknFepFnKeyManager( CAknFepManager& aFepMan, 
+                    CAknFepSharedDataInterface* aSharedDataInterface)
+:iFepMan( aFepMan ),
+ iFnKeyState( EFnKeyNone ),
+ iIsQwertyMode (EFalse),
+ iSharedDataInterface(aSharedDataInterface),
+ iShiftKeyPressed( EFalse )
+    {
+    iIsQwertyMode = iSharedDataInterface->QwertyInputMode();
+#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__
+#ifdef RD_INTELLIGENT_TEXT_INPUT
+    iWasPreviousModePredictive = iFepMan.WesternPredictive();
+#endif // RD_INTELLIGENT_TEXT_INPUT
+#else   
+    iWasPreviousModePredictive = iSharedDataInterface->PredictiveTextOn();
+#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__     
+    // The CR key KAknFepFnKeyState is reset the Fn key state NONE
+    iSharedDataInterface->SetFnKeyState(0);
+    }
+
+CAknFepFnKeyManager* CAknFepFnKeyManager::NewL( CAknFepManager& aFepMan,
+                                            CAknFepSharedDataInterface* aSharedDataInterface)
+    {
+    CAknFepFnKeyManager* self = new(ELeave) CAknFepFnKeyManager( aFepMan,aSharedDataInterface );
+    CleanupStack::PushL(self);
+    self->ConstructL();
+    CleanupStack::Pop();
+    return self;
+    }
+
+CAknFepFnKeyManager::~CAknFepFnKeyManager()
+    {    
+    iFnKeyState=EFnKeyNone;
+    }
+    
+void CAknFepFnKeyManager::ConstructL()
+    {    
+    }
+
+void CAknFepFnKeyManager::SetQwertyInputMode(TBool aQwertyState)
+    {
+    iIsQwertyMode = aQwertyState;
+    }
+void CAknFepFnKeyManager::ClearFnKeyState()
+    {
+    iFnKeyState = EFnKeyNone;
+    iFepMan.UpdateIndicators();    
+    }
+
+CAknFepFnKeyManager::TFnKeyState CAknFepFnKeyManager::FnKeyState()
+    {
+    return iFnKeyState;
+    }
+
+void CAknFepFnKeyManager::SetFnKeyState(CAknFepFnKeyManager::TFnKeyState aState)
+    {
+    TPtiKeyboardType keyboardType = iFepMan.KeyboardLayout();
+    //This function is only supported for EFnKeyNext and half Qwerty keypad at the moment.
+    if(keyboardType == EPtiKeyboardHalfQwerty && EFnKeyNext == iFnKeyState &&
+        aState == EFnKeyNone)
+        {    
+        if(aState == EFnKeyNext && keyboardType == EPtiKeyboardHalfQwerty)
+        	{
+        	iFepMan.PtiEngine()->CancelTimerActivity();
+        	}
+        iFnKeyState = aState;
+        iFepMan.SetCase( (TCase)iPreviousCase );
+        iFepMan.UpdateIndicators();
+        }
+#ifdef RD_INTELLIGENT_TEXT_INPUT  
+    else if(keyboardType == EPtiKeyboardQwerty3x11 && EFnKeyNone == iFnKeyState &&
+        aState == EFnKeyLock)
+        {
+        UpdatePreviousCase();
+        iFnKeyState = aState;
+        iFepMan.SetCase(EFnKeyLowerCase);
+        iFepMan.UpdateIndicators();
+        }
+#endif // RD_INTELLIGENT_TEXT_INPUT 		
+#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__  
+#ifdef RD_INTELLIGENT_TEXT_INPUT  
+    //Half QWERTY state uses this to change from EFnKeyNext aFnKeyNone
+ 	   else if ( iFnKeyState == EFnKeyNone && aState == EFnKeyNext )
+        {
+        // QWERTY long press implementation uses this function to set the Fn state.
+        UpdatePreviousCase();
+        iFnKeyState=EFnKeyNext;
+        }
+
+#endif // RD_INTELLIGENT_TEXT_INPUT  
+#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__
+     // TODO: above piece of code should be simplify.    
+     // Set the Fn Key state  
+ 	 else
+ 	    {
+ 	    iFnKeyState = aState;    
+ 	    }
+    }
+TKeyResponse CAknFepFnKeyManager::HandleFnKeyEventL( const TKeyEvent& aKeyEvent, TEventCode aType, 
+                                                     TCoeInputCapabilities aInputCapabilities ) 
+    {
+    
+    TKeyResponse keyResponse = EKeyWasNotConsumed;
+    TPtiKeyboardType keyboardType = iFepMan.KeyboardLayout();
+
+    if(!iIsQwertyMode)
+        {
+        return keyResponse;
+        }
+        
+  	if(!iFepMan.IsFlagSet(CAknFepManager::EFlagShiftKeyDepressed) && 
+      !iFepMan.IsFlagSet(CAknFepManager::EFlagLongShiftKeyPress)
+      && iShiftKeyPressed )
+        {
+        iShiftKeyPressed = EFalse;
+        }
+                
+    TInt caseFromFep = iFepMan.GetCurrentCase();
+    
+    TBool lockLater = EFalse;    
+    if(  EEventKeyDown == aType )
+        {
+        if( stdFnKeyCode == aKeyEvent.iScanCode )
+    		{
+            iFepMan.PtiEngine()->CancelTimerActivity();
+    		if( iPreviousKeyWasFnKey )
+    			{
+    			lockLater	= ETrue;
+    			}
+    		iPreviousKeyWasFnKey = ETrue; 			
+    		}    
+    	else
+    		{
+    		iPreviousKeyWasFnKey = EFalse; 				
+    		}        
+        }
+    if(aKeyEvent.iRepeats && stdFnKeyCode != aKeyEvent.iScanCode &&
+            iFnKeyState == EFnKeyNext)
+        {
+        // To fix the bug : Long press of number at the begin of editor. Then enter any character. 
+        // The character is shown in upper case.
+        if (ELowerCase == caseFromFep)
+            {
+            iFepMan.SetCase( (TCase)caseFromFep );
+            }
+        // To Fix Error: Character is NOT entered as per the input mode
+        // when the character is entered after pressing Shift + Fn + Shift keys
+        else if(!(EStdKeyRightShift == aKeyEvent.iScanCode ||
+        		  EStdKeyLeftShift == aKeyEvent.iScanCode))
+	        {
+            iFepMan.SetCase( (TCase)iPreviousCase );	
+            }
+            
+#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__
+#ifdef RD_INTELLIGENT_TEXT_INPUT  
+         // For QWERTY Predictive ---->
+        if(iWasPreviousModePredictive)
+        	{
+        	// We usually end up here when using the long key press to enter Fn symbols
+			iFepMan.SetWesternPredictive(ETrue); 
+		   	iFepMan.TryChangeModeL(ELatin);               	
+        	}
+        // For QWERTY Predictive <----
+
+#endif // RD_INTELLIGENT_TEXT_INPUT  
+#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__    
+        iFnKeyState = EFnKeyNone;
+        iFepMan.UpdateIndicators();
+        }
+     
+    if( stdFnKeyCode == aKeyEvent.iScanCode )
+        {
+        // Consume the function key events.
+        keyResponse = EKeyWasConsumed;
+        }
+    if (iFepMan.IsHybridAplhaEditor())
+        keyResponse = EKeyWasNotConsumed;
+ 
+    
+    // check for the case when in Half Qwerty and both shift and fn keys are
+    // pressed and fn key is released while shift is still down
+    if( EEventKeyUp == aType && 
+    	stdFnKeyCode == aKeyEvent.iScanCode &&
+    	iShiftKeyPressed &&
+    	EPtiKeyboardHalfQwerty == keyboardType )
+    	{
+    	ClearFnKeyState();
+    	}
+    	    
+    if( EStdKeyRightShift == aKeyEvent.iScanCode ||  EStdKeyLeftShift == aKeyEvent.iScanCode )
+        {
+        if( EEventKeyDown == aType ) 
+            {
+            iShiftKeyPressed = ETrue;
+            }
+        if( EEventKeyUp == aType )
+            {
+            iShiftKeyPressed = EFalse;
+            }
+        }
+        
+    // For Shift key handling.
+    // Shift Key should be ignored in locked state.
+     
+    if( ( EFnKeyNext == iFnKeyState ) 
+    && EEventKeyDown == aType &&
+    ( EStdKeyRightShift == aKeyEvent.iScanCode ||  EStdKeyLeftShift == aKeyEvent.iScanCode)) 
+        {
+        // Fnkey event+ shift key event. Change to FnKey Upper case
+        iFepMan.SetCase( EFnKeyUpperCase );
+        if( keyboardType == EPtiKeyboardHalfQwerty )
+            {
+            iFepMan.SetCase( EFnKeyLowerCase );
+            }
+        return EKeyWasNotConsumed;
+        }
+    //===================================================		
+	
+	// GQF clarification Fn+Shift
+	// If shift key is hold on, So fn key state should be EFnKeyNext
+	// So next text mode goes to function upper
+	if( aType == EEventKeyUp && iShiftKeyPressed &&  stdFnKeyCode == aKeyEvent.iScanCode)	
+		{
+		if( EFnKeyPressed == iFnKeyState)
+			{
+			iFepMan.PtiEngine()->CancelTimerActivity();
+			iFnKeyState=EFnKeyNext;
+			iFepMan.SetCase( EFnKeyUpperCase );				
+			}			
+        else if (EFnKeyPressedAgain == iFnKeyState )
+            {
+            CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyLock"));
+            iFnKeyState = EFnKeyLock;	                
+            UpdatePreviousCase();
+            iFepMan.SetCase( EFnKeyUpperCase );
+            }
+        else if(EFnKeyDown == iFnKeyState)
+            {
+            iFnKeyState = EFnKeyNone;
+            UpdatePreviousCase();
+            iFepMan.SetCase( EUpperCase );	
+            }
+		iFepMan.UpdateIndicators();
+		}
+	// Holding shift key, press key just after releasing fn key,
+	// Fn key state goes to EFnKeyNone mode.
+	if( aType == EEventKeyUp && iShiftKeyPressed && iFnKeyState==EFnKeyNext && stdFnKeyCode != aKeyEvent.iScanCode &&  !(EStdKeyRightShift == aKeyEvent.iScanCode ||  EStdKeyLeftShift == aKeyEvent.iScanCode))
+		{
+		iFnKeyState=EFnKeyNone;		
+		iFepMan.SetCase( EUpperCase );		
+		iFepMan.UpdateIndicators();
+		}
+		
+	//====================================================
+		
+    // Fnkey event+ shift key event -> when fn and shift key are kept pressed.
+    if( (iShiftKeyPressed && EFnKeyNone != iFnKeyState ) )
+        {
+        if( (EFnKeyPressed == iFnKeyState) &&
+        	(stdFnKeyCode != aKeyEvent.iScanCode) &&
+        	!( EStdKeyRightShift == aKeyEvent.iScanCode ||  EStdKeyLeftShift == aKeyEvent.iScanCode ))
+	        {
+     	   	CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyDown"));
+          	iFnKeyState=EFnKeyDown;	      	
+	        }
+        
+        // Thre is no Functin shift reverse mode
+        // So, In shift key press if fn key state is not none
+        // text case stte is function upper.
+        if( aType == EEventKeyDown && stdFnKeyCode == aKeyEvent.iScanCode)
+        	{
+        	if ( EFnKeyNext == iFnKeyState )
+        		{       	     	
+       	     	if(lockLater)
+       	     	    {
+       	     	    //CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyPressedAgain "));
+       	     	    iFnKeyState=EFnKeyPressedAgain;
+       	     	    } 					
+        		}
+        	else if ( EFnKeyLock == iFnKeyState )
+	        	{
+       	     	//CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyDown"));
+              	iFnKeyState=EFnKeyDown;
+	        	}
+        	}
+        iFepMan.SetCase( EFnKeyUpperCase );
+        if( keyboardType == EPtiKeyboardHalfQwerty )
+            {
+            iFepMan.SetCase( EFnKeyLowerCase );
+            }
+        iFepMan.UpdateIndicators();
+        return EKeyWasNotConsumed;
+        }
+
+    // Check the input mode of the currently focused editor (when available)
+    MCoeFepAwareTextEditor* fepAwareTextEditor = aInputCapabilities.FepAwareTextEditor();
+    
+    if ( fepAwareTextEditor && fepAwareTextEditor->Extension1() )
+        {
+        if( EAknEditorNumericInputMode == 
+            ( static_cast<CAknEdwinState*>( fepAwareTextEditor->Extension1()->State( KNullUid ) ) )->CurrentInputMode() )
+            // If number input mode is detected, handle it as number input
+            // capability with greatest available variation of letters
+            {
+            // Use EDialableCharacters since it provides most flexibility for number input
+            aInputCapabilities.SetCapabilities( TCoeInputCapabilities::EDialableCharacters );
+            }            
+        }
+        
+    if ( EFnKeyForced != iFnKeyState )
+        {
+        // Handling of Fn key events.
+        // If Fn key is held down while characters are typed, all following key 
+        // events are exchanged.
+        // If Fn key is only pressed (key down + key up, not held down), only
+        // the very next key event is exchanged.
+        // If the next key event after Fn key is another Fn key event, the 
+        // Fn lock mode is activated. Then all following key events are
+        // exchanged until the Fn key is pressed again.       
+         
+        if( EEventKeyDown == aType && stdFnKeyCode == aKeyEvent.iScanCode )
+            {
+            if ( EFnKeyNext == iFnKeyState )
+                {                
+                if(lockLater)
+            		{
+            		CEikonEnv::Static()->InfoMsg(_L("iFnKeyState=EFnKeyPressedAgain"));	
+            		iFnKeyState=EFnKeyPressedAgain;
+          			}
+              	else
+              	    {
+              	    iFnKeyState=EFnKeyPressed;
+              	    }		              			
+                }
+            else if ( EFnKeyLock == iFnKeyState )
+                {
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState=EFnKeyDown"));
+                iFnKeyState=EFnKeyDown;
+                }
+            //else 
+            // Sahoo
+            else if ( EFnKeyDown != iFnKeyState )
+                {
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState=EFnKeyPressed"));
+                iFnKeyState=EFnKeyPressed;
+                }
+            iFepMan.UpdateIndicators();    
+            return keyResponse;
+            }
+        // Sahoo            
+        if( EEventKeyDown == aType && EFnKeyPressed == iFnKeyState )
+            {
+            iFnKeyState=EFnKeyDown;
+            iFepMan.SetCase( EFnKeyLowerCase );
+            }
+#ifdef __WINS        
+        // NOTE: This following HACK is to ensure working in emulator environment,
+        // where Alt Gr key seems to send always two separate key events:
+        // EStdKeyRightFunc and EStdKeyLeftCtrl
+        if ( EStdKeyLeftCtrl == aKeyEvent.iScanCode )
+            {
+            return;
+            }
+        // HACK ends
+#endif
+        
+        if( EEventKeyUp == aType && stdFnKeyCode == aKeyEvent.iScanCode )
+            {
+            if ( EFnKeyPressedAgain == iFnKeyState )
+                {
+               	if(! iFepMan.IsFlagSet(CAknFepManager::EFlagQwertyShiftMode))
+	               	{
+	                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyLock"));
+	                iFnKeyState = EFnKeyLock;
+		           	}
+               	else
+	               	{
+	                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyNext"));
+	                iFepMan.PtiEngine()->CancelTimerActivity();
+	                iFnKeyState = EFnKeyNext;               		
+	               	}
+                UpdatePreviousCase();
+                iFepMan.ClearFlag(CAknFepManager::EFlagQwertyShiftMode);             
+                iFepMan.SetCase( EFnKeyLowerCase );
+                }
+            else if ( EFnKeyPressed == iFnKeyState )
+                {
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyNext"));
+                iFepMan.PtiEngine()->CancelTimerActivity();
+                iFnKeyState = EFnKeyNext;
+                
+                UpdatePreviousCase();
+                
+                // After pressing shift key Press Fn key,
+                // So its goes to function upper mode
+                if( iFepMan.IsFlagSet(CAknFepManager::EFlagQwertyShiftMode))
+                	iFepMan.SetCase( EFnKeyUpperCase );
+                else
+                    iFepMan.SetCase( EFnKeyLowerCase );
+                }
+            else
+                {
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyNone"));
+                
+                // For Shift key handling.
+                if( EFnKeyDown == iFnKeyState || EFnKeyNext == iFnKeyState )
+                    {
+                    iFepMan.SetCase( (TCase)iPreviousCase );
+                    }
+                iFnKeyState = EFnKeyNone;
+                }
+            if( EEventKeyUp == aType && iFnKeyState == EFnKeyNone )
+                {
+                // To fix the bug :Fn key lock. then type a number with short press at begin of 
+                // editor. Now press Fn key to go to Fn none state. 
+                // Press a character. The char is in upper case
+                if (ELowerCase == caseFromFep)
+		            {
+		            iFepMan.SetCase( (TCase)caseFromFep );
+		            }
+		        // To Fix Error: Character is NOT entered as per the input mode
+		        // when the character is entered after pressing Shift + Fn + Shift keys
+		        else if(!(EStdKeyRightShift == aKeyEvent.iScanCode ||
+		            	  EStdKeyLeftShift == aKeyEvent.iScanCode))
+		            {
+		            iFepMan.SetCase( (TCase)iPreviousCase );	
+		            }
+                }
+            iFepMan.UpdateIndicators();
+            return keyResponse;
+            }
+        }
+                
+    // Now detect the input mode for telephone and calculator apps
+/*    if ( aInputCapabilities.IsNone() )
+        {   // Idle and Calculator do not provide input capabilities at all.
+        if ( KPhoneUidAppPhone == aCurrentTopAppUid || KUidIdle == aCurrentTopAppUid )
+            {
+            aInputCapabilities.SetCapabilities( TCoeInputCapabilities::EDialableCharacters );
+            }
+        if ( aCurrentTopAppUid == KUidCalc )
+            {
+            aInputCapabilities.SetCapabilities( TCoeInputCapabilities::EWesternNumericReal );
+            }
+        }        
+*/        
+    // Now handle the automatic number mode
+    if ((iFnKeyState != EFnKeyForced) && 
+        ( aInputCapabilities.SupportsWesternNumericIntegerPositive() ||
+         aInputCapabilities.SupportsWesternNumericIntegerNegative() ||
+         aInputCapabilities.SupportsWesternNumericReal() ||
+         aInputCapabilities.SupportsDialableCharacters() )         
+       )
+        {
+        // If input mode is numeric, Fn key is automated (=forced)
+        iFnKeyState = EFnKeyForced;
+        
+        UpdatePreviousCase();
+        
+        iFepMan.SetCase( EFnKeyLowerCase );
+        iFepMan.UpdateIndicators();
+        //Forced input mode
+        // The CR key KAknFepFnKeyState set to Locked
+        iSharedDataInterface->SetFnKeyState(EFnKeyForced/*=6*/);        
+        CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyForced"));   
+        }
+#ifdef __REVERSE_FN_KEY_SUPPORTED
+    //Reverse Fn Key mapping!
+    //Just one of the cases for Reverse Fn key mapping!
+    // There are manu more of these!
+    if((EFnKeyForced == iFnKeyState && aKeyEvent.iModifiers & EModifierRightFunc ) )
+    	{    	
+    	iFepMan.SetCase( (TCase)iPreviousCase );    	
+    	iFepMan.SetReverseFnkeyInputMode(ETrue);    	
+    	}
+    else if(iFepMan.IsReverseFnkeyInput())
+    	{
+    	iFepMan.SetReverseFnkeyInputMode(EFalse);           
+        iFepMan.SetCase(EFnKeyLowerCase);
+    	}
+#endif //__REVERSE_FN_KEY_SUPPORTED
+        
+    // If any of the following states apply, the key event should be exchanged.        
+    if ( EFnKeyPressed == iFnKeyState ||
+         EFnKeyDown == iFnKeyState ||
+         EFnKeyNext == iFnKeyState ||
+         EFnKeyPressedAgain == iFnKeyState ||
+         EFnKeyLock == iFnKeyState ||
+         EFnKeyForced == iFnKeyState )
+        {
+        // For telephone number editor key events for w, p and + must be accepted as is
+        if( aInputCapabilities.SupportsDialableCharacters() && ( 'W' == aKeyEvent.iScanCode ||
+                                                                 'P' == aKeyEvent.iScanCode ||
+                                                                 '+' == aKeyEvent.iScanCode ||
+                                                                 'w' == aKeyEvent.iCode ||
+                                                                 'p' == aKeyEvent.iCode ||
+                                                                 '+' == aKeyEvent.iCode) )
+            {
+            return keyResponse;
+            }
+            
+        if( EEventKeyUp == aType )
+            {
+            if ( EFnKeyPressed == iFnKeyState || EFnKeyPressedAgain == iFnKeyState )
+                {
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyDown"));
+                iFnKeyState = EFnKeyDown;                
+                UpdatePreviousCase();
+                
+                // Fn key hold on and press any key
+                if( iFepMan.IsFlagSet(CAknFepManager::EFlagQwertyShiftMode))
+                	iFepMan.SetCase( EFnKeyUpperCase );
+                else
+                	iFepMan.SetCase( EFnKeyLowerCase );
+                }
+                
+             // Debasish MT error fixing
+            else if( EFnKeyNext == iFnKeyState && iFepMan.IsFlagSet(CAknFepManager::EFlagQwertyShiftMode))
+	            {
+	            // keep the sate as it is
+	            UpdatePreviousCase();	
+	            iFepMan.SetCase( EFnKeyUpperCase );
+	            
+	            if( keyboardType == EPtiKeyboardHalfQwerty )
+                    {
+                    iFepMan.SetCase( EFnKeyLowerCase );
+                    }
+	            }
+            else if ( EFnKeyNext == iFnKeyState )
+                {
+                //This has to be done if the keyboard type is not half Qwerty
+                if(keyboardType == EPtiKeyboardHalfQwerty)
+                    return keyResponse;
+                CEikonEnv::Static()->InfoMsg(_L("iFnKeyState = EFnKeyNone"));
+                
+                // To fix the bug : At begin of editor, press Fn key. Now press a number.
+                // Now state is FnKeyNone. Type any char. The case is upper.
+                // For Shift key handling.
+                if( EFnKeyDown == iFnKeyState || EFnKeyNext == iFnKeyState )
+                    {
+		 			if (ELowerCase == caseFromFep)
+	            		{
+	            		iFepMan.SetCase( (TCase)caseFromFep );
+	            		}
+			        // To Fix Error: Character is NOT entered as per the input mode
+			        // when the character is entered after pressing Shift + Fn + Shift keys
+		 			else if(!(EStdKeyRightShift == aKeyEvent.iScanCode ||  
+		 						  EStdKeyLeftShift == aKeyEvent.iScanCode))
+	            		{
+	            		iFepMan.SetCase( (TCase)iPreviousCase );	
+	            		}
+                    }
+                 iFnKeyState = EFnKeyNone;
+                }
+            iFepMan.UpdateIndicators();
+            }                        
+        }
+
+    return keyResponse;
+    }
+  
+void CAknFepFnKeyManager::UpdatePreviousCase()
+    {
+    iPreviousCase = iFepMan.GetCurrentCase();
+    }
+// End of File
+