fep/aknfep/src/AknFepKeyCatcher.cpp
author Shabe Razvi <shaber@symbian.org>
Thu, 02 Sep 2010 15:52:50 +0100
branchRCL_3
changeset 45 6f51b41715c8
parent 44 ecbabf52600f
child 50 5a1685599b76
permissions -rw-r--r--
Merge RCL_3 fixes with reverted delivery

/*
* Copyright (c) 2002-2004 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 the CAknFepKeyCatcher methods.
*
*/












#include <eikenv.h>
#include <eikappui.h>               //iEikonEnv
#include <e32keys.h>                //keys
#include <uikon.hrh>                //keys

#include "AknFepKeyCatcher.h"
#include "AknFepManager.h"
#include "AknFepGlobalEnums.h"
#include "AknFepPanic.h"
#include "AknFepPluginManager.h"
#include <AknDef.h>
#include <PtiEngine.h>
#include <PtiDefs.h>                //keys
#include <featmgr.h>                //FeatureManager
#include <AvkonInternalCRKeys.h>

LOCAL_C void FepObserverHandleCompletionOfTransactionL(MCoeFepObserver& aFepObserver)
    {
    aFepObserver.HandleCompletionOfTransactionL();
    }

CAknFepKeyCatcherInterface::~CAknFepKeyCatcherInterface()
    {
    }

void CAknFepKeyCatcherInterface::HandleChangeInFocus()
    {
    }

CAknFepKeyCatcher* CAknFepKeyCatcher::NewL(CAknFepManager& aFepMan)
    {
    CAknFepKeyCatcher* self=new(ELeave) CAknFepKeyCatcher(aFepMan);
    CleanupStack::PushL(self);
    self->ConstructL();
    CleanupStack::Pop(); //self
    return self;
    }

CAknFepKeyCatcher::~CAknFepKeyCatcher()
    {
    CloseWindow();
    if(NULL != iEikonEnv->EikAppUi())
        {
        iEikonEnv->EikAppUi()->RemoveFromStack(this);
#ifdef RD_SCALABLE_UI_V2
        (CEikonEnv::Static())->RemoveMessageMonitorObserver(*this); 
#endif //RD_SCALABLE_UI_V2
        }
    }

TKeyResponse CAknFepKeyCatcher::OfferKeyEventL(const TKeyEvent& aKeyEvent, TEventCode aEventCode)
    {
    CheckForSpecilacaseHandlingTosetKeyboardtype(aKeyEvent, aEventCode);
#ifdef RD_SCALABLE_UI_V2     
    if (aEventCode == EEventKeyUp && 
        aKeyEvent.iScanCode == EStdKeyNo)
        {
        iFepMan.HandleEndKeyL();
        }
#endif //RD_SCALABLE_UI_V2 
    // Temporary solution for Disable key tone when press key in Virtual ITUT
    iFepMan.NeedDisableKeySound(aKeyEvent, aEventCode);
    // sort out if we're doing long or short presses...
    TKeyPressLength length = aKeyEvent.iRepeats ? ELongKeyPress : EShortKeyPress;
    if (iKeyCatcherState == EAknFepStateNull)
        {
        if (aEventCode == EEventKeyUp && iLongPressedScanCode == aKeyEvent.iScanCode)
            {
            // An editor has lost focus during long key press (for example SCT popup 
            // is shown during long keypress of star key).
            // EFlagLongKeyPressHandled needs to be reseted also in this case.
            ClearFlag(EFlagLongKeyPressHandled);
            iLongPressedScanCode = 0;
            }
        if((aEventCode == EEventKeyDown) && (aKeyEvent.iScanCode==EStdKeyRightShift) 
                && (iFepMan.KeyboardLayout()== EPtiKeyboard12Key) && !iFepMan.Japanese() 
                &&  !iFepMan.IsChineseInputLanguage()  )
            {
            iFepMan.SetCcpuFlag(CAknFepManager::ECcpuStateShiftkeyWasPressedBeforeLosingFocus);
            }
        else if ((aEventCode == EEventKeyUp) && (aKeyEvent.iScanCode==EStdKeyRightShift) 
                && (iFepMan.KeyboardLayout()== EPtiKeyboard12Key) 
                && iFepMan.IsCcpuFlagSet(CAknFepManager::ECcpuStateShiftkeyWasPressedBeforeLosingFocus))
            {
            iFepMan.ClearCcpuFlag(CAknFepManager::ECcpuStateShiftkeyWasPressedBeforeLosingFocus);
            }
        
        TInt keyCode = aKeyEvent.iCode;
        if (keyCode == EKeyF19 || keyCode == EKeyF22 ||
            keyCode == EKeyF23 || keyCode == EKeyF24 )
            {
            // Internal FEP key events that are simulated by FEP are always 
            // also consumed by FEP. In this situation FEP has simulated the
            // key event but the editor is unfocused right after.
            return EKeyWasConsumed;
            }
        // FEP does not handle any keyevents if there are not any FEP aware editor focused, except shift key. 
        // Notice that shift key is handled only if CAknFepKeyCatcher is constructed by focusing some editor.
        
        if (iFepMan.HashKeySelectionInUse())
        	{        	
        	// Hash key selection should work also with uneditable editors, therefore
        	// we need to fake it as a shift key press here.
	        if (aKeyEvent.iScanCode == EStdKeyHash)
    	    	{
    	    	if (aEventCode == EEventKeyUp)
    	    		{		
    	    		if (iFepMan.IsFlagSet(CAknFepManager::EFlagLongShiftKeyPress))	   			    	    		
    	    			{    	    			    	    		
    	    			// We need to generate real shift up event for editor to
    	    			// cancel ccpu-mode.
		   				TKeyEvent ccpuStart = {0, EStdKeyLeftShift, 0, 0};
    	 				CCoeEnv::Static()->SimulateKeyEventL(ccpuStart, aEventCode);
    	    			}
    	    		else
    	    			{
    	    			// CCpu-mode was not yet activated. Just cancel the timer.
    	    			iFepMan.CancelShiftKeyTimer();
    	    			}		
    	 			return EKeyWasNotConsumed;       	    		
    	    		}
				else if (aEventCode == EEventKeyDown)   	    		
					{
					return iFepMan.HandleShiftKeyEventL(aEventCode);        							
					}
        		}
    	    }
    	
    	if( keyCode == EKeyEscape )
    		{
    		return iFepMan.HandleKeyEventL(keyCode, length);	
    		}
    		
        if ((aKeyEvent.iScanCode==EStdKeyRightShift || aKeyEvent.iScanCode==EStdKeyLeftShift))
            {
            if (aEventCode == EEventKeyUp)
                {
                iFepMan.SetExtendedFlag(CAknFepManager::EExtendedFlagShiftReleasedOnPopup);
                }
            else
                {
                iFepMan.ClearExtendedFlag(CAknFepManager::EExtendedFlagShiftReleasedOnPopup);
                }
            }
        else
            {
            return EKeyWasNotConsumed;
            }
        }
#ifdef RD_INTELLIGENT_TEXT_INPUT
     if(iKeyCatcherState != EAknFepStateNull)
        {
            if(iFepMan.IsFnKeyMapped() &&
                (iFepMan.HandleFnKeyEventL( aKeyEvent, aEventCode ) == EKeyWasConsumed))
            {
            return EKeyWasConsumed;        
            }

        }
#endif
	
     if ( aEventCode == EEventKeyUp && 
          aKeyEvent.iScanCode == EStdKeyHash &&
          iFepMan.IsExtendedFlagSet(CAknFepManager::EExtendedFlagShortPressHashKey) )
         {
         // We are here because of short press of has key. Since, we had consumed
         // short press of hash key, we re-simulate is again, so that the respective
         // application can handle it as per their need.
         
         TKeyEvent keyHash = {EPtiKeyHash, EStdKeyHash, 0, 0};
         CCoeEnv::Static()->SimulateKeyEventL(keyHash, EEventKey);
         iFepMan.ClearExtendedFlag(CAknFepManager::EExtendedFlagShortPressHashKey);
         }
    // Check for the Wireless Keyboard Modifier Flag.
    const TUint KModifierExternalKeyboard = 0x00200000;
    if ( aKeyEvent.iModifiers & KModifierExternalKeyboard )
        {        
        if ( iFepMan.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction |
             CAknFepManager::EFlagInsideMultitapInlineEditingTransaction) ) 
            {
            if (iFepMan.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction))
                {
                // Remove any no matching word indicator.
                iFepMan.TryRemoveNoMatchesIndicatorL();
                }
            // Ensure the CBA is in the default state.
            iFepMan.UpdateCbaL(NULL);
            // Close FEP UI, commit the inline edit and show the cursor.            
            iFepMan.TryCloseUiL(); 
            }        
        // Allow the key through to the underlying control.
        return EKeyWasNotConsumed;
        }
    
    TKeyResponse retCode;
	if (iFepMan.HandleCcpuModeKeyEventL(aKeyEvent, aEventCode,
	                                   retCode, IsFlagSet(EFlagLongKeyPressHandled)))
		{
		return retCode;	
		}

#ifdef RD_SCALABLE_UI_V2
    // Set text selection when tap arrow keys
	if(iFepMan.HandleSelModeArrowKeyEventL(aKeyEvent, aEventCode, retCode))
        {
        return retCode;
        }        
#endif //RD_SCALABLE_UI_V2	

    
  
    if ( ( aKeyEvent.iModifiers & EModifierCtrl
#ifdef RD_INTELLIGENT_TEXT_INPUT     
           || aKeyEvent.iScanCode==EStdKeyRightCtrl
           || aKeyEvent.iScanCode==EStdKeyLeftCtrl
#endif
		 )
        && aKeyEvent.iScanCode != EStdKeyBackspace  
        && aKeyEvent.iScanCode != EStdKeyDelete )
        {// Chinese qwerty increment input mode by Ctrl+Shift
		return iFepMan.HandleQwertyControlKeyEventL(aKeyEvent, aEventCode);        
        }

    // handling hash key event in japanese input
    if (iFepMan.IsFeatureSupportedJapanese()
     && aKeyEvent.iScanCode==EStdKeyHash && !iFepMan.IsOnlyNumericPermitted())
        {
        return HandleHashKeyJapaneseL(aKeyEvent, aEventCode);
        }
    
    // Let FepMan handle chr key
    if (aKeyEvent.iScanCode==EStdKeyLeftFunc)
        {
        if (iFepMan.HandleQwertyChrKeyEventL(aEventCode))
            {
            return EKeyWasConsumed;
            }
        }
    //let FepMan handle shifts
    if ( aKeyEvent.iScanCode==EStdKeyRightShift || aKeyEvent.iScanCode
            ==EStdKeyLeftShift )
        {
        // phrase creation
        if ( iFepMan.IsPinyinPhraseCreation() )
            {
            return EKeyWasConsumed;
            }
        else
            {
            return iFepMan.HandleShiftKeyEventL( aEventCode );
            }
        }
    
    if (iFepMan.Japanese())
        {
        if ((aKeyEvent.iScanCode==EStdKeyDevice0) || (aKeyEvent.iScanCode==EStdKeyDevice1))
            {
            if (aEventCode != EEventKey)
                {
                return EKeyWasNotConsumed;
                }
            }
        }

    if (aEventCode==EEventKeyUp)
        {
        if ( iLongPressedScanCode == aKeyEvent.iScanCode )
            {
            ClearFlag(EFlagLongKeyPressHandled);
            iLongPressedScanCode = 0;
            }
        
        if ( (EStdKeyDelete == aKeyEvent.iScanCode || 
              EStdKeyBackspace == aKeyEvent.iScanCode))  
                {
                iFepMan.SetLongClearAfterCloseUI(EFalse);               
                }
            
        iFepMan.ClearFlag(CAknFepManager::EFlagCharacterAdded);
        }


    //let FepMan to try to handle Thai 0 key
    TBool thai0Keyhandling = EFalse;

    if (aKeyEvent.iScanCode==EPtiKey0 
        && (!(aKeyEvent.iModifiers & EModifierSpecial)  // If the key is produced by hardware key
          // In VKB and FSQ latin range 0 could be input with virtual keyboard
          || (iFepMan.PluginInputMode() != EPluginInputModeFSQ  
             && iFepMan.PluginInputMode() != EPluginInputModeVkb)))
        {
        if (iFepMan.HandleThai0KeyEvent(aEventCode, length, thai0Keyhandling) == EKeyWasConsumed)
            {
            return (EKeyWasConsumed); // Eats some key events to find out when o key goes up
            }
        if (thai0Keyhandling)
            {
            if(length == ELongKeyPress)
                {
                SetFlag(EFlagLongKeyPressHandled);
                iLongPressedScanCode = aKeyEvent.iScanCode;
                }
            return (iFepMan.HandleKeyEventL(EPtiKey0, length));
            }
        }
    TInt keyCode = aKeyEvent.iCode;
    TKeyResponse response = EKeyWasNotConsumed;
    
#ifdef RD_INTELLIGENT_TEXT_INPUT
    if ((aKeyEvent.iScanCode == EStdKeyNkpAsterisk ||aKeyEvent.iScanCode == EPtiKeyStar )&& iFepMan.KeyboardLayout() == EPtiKeyboard12Key)
    	{
    	//For Special Editors like Java numeric editor, EventKeyUp is allowed to flow through FEP.
    	//All other events for Special Editor is not consumed by FEP.
    	if(iFepMan.IsSpecialNumericEditor() && iFepMan.IsOnlyNumericPermitted() && aEventCode != EEventKeyUp)
    		{
    		return EKeyWasNotConsumed;
    		}
#ifdef FF_HOME_SCREEN_EASY_DIALING
				// If Easy Dialing Feature in Phone Idle is turned on then,
				// phoneIdleinEasyDialingNumericMode is set to True
        TBool phoneIdleinEasyDialingNumericMode = EFalse;
        phoneIdleinEasyDialingNumericMode = 
        	(iFepMan.EditorType() == CAknExtendedInputCapabilities::EPhoneNumberEditor) &&
        	(iFepMan.InputMode() == ENumber || iFepMan.InputMode() == ENativeNumber);
#endif           
    	// For all Numeric Editors, only Event Key is allowed to flow through
    	// When Easy Dialing is on, the editor in not numeric, even though in this scenario only Event key is allowed throug FEP
    	if ( (iFepMan.IsOnlyNumericPermitted()
#ifdef FF_HOME_SCREEN_EASY_DIALING    	
    		 || phoneIdleinEasyDialingNumericMode 
#endif    		 
    		 )&& 
    	     aEventCode != EEventKey)
    		{
    		return EKeyWasConsumed;
    		}
    	
    	/* 
    	For Long Key press:
    		We only have Event Key, and FEP handles it for launching SCT.
    	For short key presses:
    		For Non numeric editors:
    			EventDown and Eventkey:
    				FEP consumes it - if FEP is in Inline edit or it can launch SCT. It is consumed by below check and does not flow further.
    				Else, these events flow through FEP. Later decision is made to consume the event or not.
    			EventUpKey:
    				FEP consumes it - if FEP is in Inline edit or it can launch SCT. Desired functionality is achieved after flowing through FEP.
    				Else, these events flow through FEP. Later decision is made to consume the event or not.
    		For Numeric Editors:
    			FEP handles EventKey for star key looping or launching sct. Other events does not reach here.
    	*/
    	if ( length != ELongKeyPress && 
    	     aEventCode != EEventKeyUp && 
    	     (
    	      iFepMan.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction) ||
    	      !iFepMan.IsAbleToLaunchSCT() 
    	     ) &&
    	     (!iFepMan.IsOnlyNumericPermitted()
#ifdef FF_HOME_SCREEN_EASY_DIALING    	
    		  && !phoneIdleinEasyDialingNumericMode 
#endif    		     	     
    	     )
    	   )
    		{
            return EKeyWasConsumed;
    		}
    	keyCode = EPtiKeyStar;
    	}
    else
#endif
    //only interested in EventKeys apart from for the shift key
    if (aEventCode!=EEventKey)
        {
        // For Japanese, don't pass EEventKeyUp and EEventKeyDown
        // event to application side if we are inputting Hiragana/Kanji.
        if (iFepMan.Japanese() && iKeyCatcherState == EAknFepStateUIActive)
            {
            // arrow keys
            if (aKeyEvent.iScanCode == EStdKeyDownArrow
                  || aKeyEvent.iScanCode == EStdKeyRightArrow
                  || aKeyEvent.iScanCode == EStdKeyLeftArrow
                  || aKeyEvent.iScanCode == EStdKeyUpArrow)
                {
                return EKeyWasConsumed;
                }
            }

        if ((iKeyCatcherState == EAknFepStateInitial)
            && (aEventCode == EEventKeyUp) && (aKeyEvent.iScanCode == EStdKeyBackspace))
            {
            CCoeEnv::Static()->ForEachFepObserverCall(FepObserverHandleCompletionOfTransactionL);
            }

        return EKeyWasNotConsumed;  
        }

    //if we've done a long press, and we get repeats, don't pass it on to the fep man.

    if (!(iFepMan.Japanese()&&
          (aKeyEvent.iCode == EKeyLeftArrow
        || aKeyEvent.iCode == EKeyRightArrow
        || aKeyEvent.iCode == EKeyDownArrow
        || aKeyEvent.iCode == EKeyUpArrow)))
        {
        if (!(aKeyEvent.iCode == EKeyPrevious || aKeyEvent.iCode == EKeyNext))
            {                	
            if (IsFlagSet(EFlagLongKeyPressHandled)&&(aKeyEvent.iRepeats))
                {
                return EKeyWasConsumed;         
                }
            }
        }

#ifdef RD_INTELLIGENT_TEXT_INPUT
    CAknFepFnKeyManager::TFnKeyState fnKeyStateBeforeKey = iFepMan.FnKeyState();

	// the short key press was blocked, don't allow the long key press to go through
	if ( IsFlagSet(EFlagBlockAllLongKeyPressEvents) && length == ELongKeyPress 
								&& !IsNaviKey(aKeyEvent.iCode) )
		{
		return EKeyWasNotConsumed;
		}
	// if the editor has no free space	
	// the key press is a valid one 
	// (it is not for eg. a simulated key press for case change)
	// we are not in between multitapping
	// block the subsequent long key press also	
	if ( length == EShortKeyPress && !iFepMan.EditorHasFreeSpace() 
		&& iFepMan.PtiEngine()->IsValidKey((TPtiKey)aKeyEvent.iScanCode)   
		&& !iFepMan.IsFlagSet(CAknFepManager::EFlagInsideMultitapInlineEditingTransaction) ) 
    	{
    	SetFlag(EFlagBlockAllLongKeyPressEvents); // block the subsequent long key press also
    	return EKeyWasNotConsumed;
    	}
    else
    	{
    	ClearFlag(EFlagBlockAllLongKeyPressEvents);
    	}	
#endif // RD_INTELLIGENT_TEXT_INPUT 
    if (iFepMan.HandleQwertyKeyEventL(aKeyEvent, response))
        {

#ifdef RD_INTELLIGENT_TEXT_INPUT
#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__ 
		ClearFlag(EFlagBlockAllLongKeyPressEvents);	
#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__    
			TPtiKeyboardType keyboardType = iFepMan.KeyboardLayout();
	        if( (length == ELongKeyPress) &&
            (
#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__ 
            iFepMan.LongPressNumberEntryOnQwerty()||
#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__          
            (keyboardType == EPtiKeyboardHalfQwerty)) &&
            iFepMan.KeyMapsDifferentCharacterWithFn( (TPtiKey)aKeyEvent.iScanCode ) && 
            fnKeyStateBeforeKey == CAknFepFnKeyManager::EFnKeyNone )
            {
            SetFlag(EFlagLongKeyPressHandled); // don't allow any more long presses to go through.
            iLongPressedScanCode = aKeyEvent.iScanCode;
            }
#endif // RD_INTELLIGENT_TEXT_INPUT      
        return response;
        }
#ifdef RD_INTELLIGENT_TEXT_INPUT		
#ifdef __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__ 
        // key was not handled because there is no space in editor
    else if( !iFepMan.EditorHasFreeSpace() && ( iFepMan.KeyboardLayout() != EPtiKeyboard12Key))
    	{		
    	SetFlag(EFlagBlockAllLongKeyPressEvents);
    	}
#endif // __ITI_LONGPRESS_NUM_SHIFT_COPYPASTE__          			
#endif // RD_INTELLIGENT_TEXT_INPUT    
	if (iFepMan.IsFlagSet(CAknFepManager::EFlagShiftKeyDepressed))
		{
		// If shift (ie. edit-key) is pressed in itu-t mode, block all the numeric keys, 
		// otherwise they will be routed to edwin and prodec unwanted character.
		if ((aKeyEvent.iScanCode >= EPtiKey1 && aKeyEvent.iScanCode <= EPtiKey9) ||
		    aKeyEvent.iScanCode == EPtiKey0 || aKeyEvent.iScanCode == EStdKeyFullStop )
			{
			return EKeyWasConsumed;
			}		
		}        
		
	if (iFepMan.IsFlagSet(CAknFepManager::EFlagQwertyChrKeyDepressed))
		{
		if ((aKeyEvent.iScanCode >= EPtiKeyQwertyA && aKeyEvent.iScanCode <= EPtiKeyQwertyZ) ||
		     aKeyEvent.iScanCode == EStdKeyFullStop )
			{
			return EKeyWasConsumed;
			}		
		}        
        
    // bulk of the selection.
    switch(keyCode)
        {
    case EKeyLeftArrow: //fall through
    case EKeyRightArrow:
    case EKeyDownArrow:
    case EKeyUpArrow:
        response = iFepMan.HandleKeyEventL(keyCode, length);
        break;
    case EKeyCBA1:
        {
        TInt inputMode = iFepMan.InputMode();
        if ( iKeyCatcherState == EAknFepStateUIActive && 
            inputMode >= ECangJie && inputMode <= EStroke )
            {
            // Left softkey does not have functionality in Chinese input modes when 
            // FEP UI is active.
            // phrase creation
            iFepMan.HandleKeyEventL( keyCode, length );
            response = EKeyWasConsumed;
            break;
            }
        }
        //fall through
    case EKeyOK:
    case EKeyCBA2:
    case EPtiKey0:
    case EPtiKey1:
    case EPtiKey2:
    case EPtiKey3:
    case EPtiKey4:
    case EPtiKey5:
    case EPtiKey6:
    case EPtiKey7:
    case EPtiKey8:
    case EPtiKey9:
    case EPtiKeyStar:
    case EPtiKeyHash:
    case EKeyEscape:
        if(length == ELongKeyPress)
            {
            SetFlag(EFlagLongKeyPressHandled); // don't allow any more long presses to go through.
            iLongPressedScanCode = aKeyEvent.iScanCode;
            }
        //fall through
    case EKeyBackspace:
    case EKeyDelete:
        
        // This check is added to handle the case when user enters a consonant
        // followed by a halant, follwed by digit using long key press.
        // When backspace is done, first time, the digit gets deleted.
        // Second time, only halant should be deleted. But since ZWS is
        // present, the entire syllable gets deleted. Hence we forcefully
        // remove the ZWS.
        
        if( keyCode == EKeyBackspace && 
            iFepMan.PreviousChar( ETrue ) == ( ZERO_WIDTH_SPACE ) )
            {
            // Remove the Zero Width Space
	    	iFepMan.RemovePreviousCharacterL();
            }

#ifdef RD_MARATHI
		if( ( keyCode == EKeyBackspace ) && 
			iFepMan.PreviousChar( ETrue ) == ( ZERO_WIDTH_JOINER ))
			{
			// Remove the Zero Width Joiner
			iFepMan.RemovePreviousCharacterL();
			}
#endif // RD_MARATHI
#ifdef RD_INTELLIGENT_TEXT_INPUT
		if( keyCode == EKeyBackspace && 
			( (aKeyEvent.iModifiers & EModifierLeftShift) ||
			  (aKeyEvent.iModifiers & EModifierRightShift) ||
			  (aKeyEvent.iModifiers & EModifierShift) ))
			{
			keyCode = EKeyDelete;
			}
#endif //RD_INTELLIGENT_TEXT_INPUT			
        response = iFepMan.HandleKeyEventL(keyCode, length, aEventCode );
        break;
    case EKeyF19: // Fep simulated event to asynchronously update the case.
    case EKeyF22:
    case EKeyF23:
    case EKeyF24:
        response = iFepMan.HandleKeyEventL(keyCode, length);
        break;
    case EKeyPowerOff:
    case EKeyPhoneEnd:
    case EKeyApplication:
#ifndef RD_INTELLIGENT_TEXT_INPUT
    		// To rip off suggested word completion when user press END / APPLICATION
    			if(iFepMan.IsAutoCompleteOn())
    			{
    			iFepMan.RemoveSuggestedCompletionL();	
    			}
#endif //RD_INTELLIGENT_TEXT_INPUT
#ifdef RD_INTELLIGENT_TEXT_INPUT
		response = iFepMan.HandleKeyEventL(keyCode, length);
#endif //RD_INTELLIGENT_TEXT_INPUT
        break;
    case EKeyPhoneSend:
        // handling phone send key event in japanese input
        if (iFepMan.IsFeatureSupportedJapanese())
            {
            if (iFepMan.Japanese() && iKeyCatcherState == EAknFepStateUIActive)
                {
                // Ensure the CBA is in the default state.
                iFepMan.UpdateCbaL(NULL);
                // Close FEP UI, commit the inline edit and show the cursor.            
                iFepMan.TryCloseUiL();
                }
            }
        break;
    default:
        break;
        }
   
    return response;
    }

void CAknFepKeyCatcher::SetState(const enum TAknFepManagerState aState)
    {
    iKeyCatcherState=aState;
    }

CAknFepKeyCatcher::CAknFepKeyCatcher(CAknFepManager& aFepMan)
    : iFepMan(aFepMan),
      iKeyCatcherState(EAknFepStateNull),iPreviousKeyeventNormal(ETrue),
      iPreviousKeyeventSpecial(EFalse),iRetainKeyBoardtypeFornextkeyevent(EFalse)
    {
    }

void CAknFepKeyCatcher::ConstructL()
    {
    // Set up fep key catching control - front window, null size, non-focusing
    CreateWindowL();
    SetFocusing(EFalse);    
    RWindow& window=Window();
    window.SetOrdinalPosition(0, ECoeWinPriorityFep);
    TPoint fepControlPos(0, 0);
    SetExtent(fepControlPos, TSize(0,0));               //null size
    window.SetExtent(fepControlPos, TSize(0,0));
    window.SetNonFading(ETrue);
    if(NULL == iEikonEnv->EikAppUi())
        {
        User::Leave(KErrCancel);
        }
    iEikonEnv->EikAppUi()->AddToStackL(this, ECoeStackPriorityFep, 
                                       ECoeStackFlagRefusesFocus|ECoeStackFlagSharable);
#ifdef RD_SCALABLE_UI_V2
    (CEikonEnv::Static())->AddMessageMonitorObserverL(*this); 
#endif  //RD_SCALABLE_UI_V2
    SetBlank();
    }

TKeyResponse CAknFepKeyCatcher::HandleHashKeyJapaneseL(const TKeyEvent& aKeyEvent, 
                                                       TEventCode aEventCode)
    {
    // handling hash keypress event only
    __ASSERT_DEBUG(aKeyEvent.iScanCode==EStdKeyHash, AknFepPanic(EAknFepPanicNotSupportKey));

    // No handling case
    if (aEventCode == EEventKeyDown
     || iKeyCatcherState == EAknFepStateNull)
        {
        if (aEventCode == EEventKeyUp)
            {
            ClearFlag(EFlagLongKeyPressHandled);
            }
        return EKeyWasNotConsumed;
        }
    
    // sort out if we're doing long or short presses...
    TKeyPressLength length = aKeyEvent.iRepeats ? ELongKeyPress : EShortKeyPress;

    // long keypress is handled once, after key event always is consumed.
    if (aEventCode == EEventKey && IsFlagSet(EFlagLongKeyPressHandled) )
        {
        return EKeyWasConsumed;
        }
    
    if (length == ELongKeyPress && !IsFlagSet(EFlagLongKeyPressHandled))
        {
        SetFlag(EFlagLongKeyPressHandled);
        }
    
    // Always handling case
    if (aEventCode != EEventKeyUp && IsFlagSet(EFlagLongKeyPressHandled)
        || aEventCode == EEventKeyUp && !IsFlagSet(EFlagLongKeyPressHandled))
        {
        TKeyResponse response = iFepMan.HandleKeyEventL(EPtiKeyHash, length);
        __ASSERT_DEBUG(response != EKeyWasNotConsumed, AknFepPanic(EAknFepPanicNotHandleHashKey));
        }
    else if (aEventCode == EEventKey
          && iKeyCatcherState == EAknFepStateUIActive)
        {
        // Ensure the CBA is in the default state.
        iFepMan.UpdateCbaL(NULL);
        // Close FEP UI, commit the inline edit and show the cursor.
        iFepMan.TryCloseUiL();
        }

    if (aEventCode == EEventKeyUp)
        {
        ClearFlag(EFlagLongKeyPressHandled);
        }
    
    return EKeyWasConsumed;
    }

void CAknFepKeyCatcher::HandleResourceChange(TInt aType)
    {
    if(aType == KEikDynamicLayoutVariantSwitch)  //If the layout has been changed, notify fep manager
        {
        iFepMan.HandleResourceChange(aType);
        }

    CCoeControl::HandleResourceChange(aType);
    }
TBool CAknFepKeyCatcher::IsNaviKey(TUint aCode)
	{
	switch(aCode)
		{
			case EKeyBackspace:
			case EKeyLeftArrow:
			case EKeyRightArrow:
			case EKeyUpArrow:
			case EKeyDownArrow:
			case EKeyDelete:
				return ETrue;
			default:
				return EFalse;	
		}
	}
#ifdef RD_SCALABLE_UI_V2

void CAknFepKeyCatcher::MonitorWsMessage(const TWsEvent& aEvent)
    {
	switch (aEvent.Type())
		{
	    case EEventPointer:
            TRAP_IGNORE(iFepMan.HandlePointerEventL(*aEvent.Pointer()));
		    break;
		default:
		    break;
		}
    }

#endif //RD_SCALABLE_UI_V2 

//To set to ITU-T mode when qwerty keypad is open and user try to use ITU-T Keypad.
TKeyResponse CAknFepKeyCatcher::CheckForSpecilacaseHandlingTosetKeyboardtype(const TKeyEvent& aKeyEvent, TEventCode aEventCode)
	{
	
	//These modifiers will set by dialer application, to simulate
	// touch button events, hence check event has come from touch keypad and set keyboardlayout accordingly
			
	if(aKeyEvent.iModifiers & EModifierNumLock &&
        aKeyEvent.iModifiers & EModifierKeypad 
        )
		{
		//Check for modifiers and previous event handled is normal or special.
		if(iFepMan.KeyboardLayout() != EPtiKeyboard12Key &&
			 iPreviousKeyeventNormal)
	        {	        
	        iFepMan.SetKeyboardLayout(EPtiKeyboard12Key);
	        iFepMan.SetFnKeyMappingState();
	        iFepMan.SetQwertyMode(0);
	        
	        //This variable will be set, while handling special key events,	        
	        iPreviousKeyeventSpecial = ETrue;
	        
	        //This variable will be set while handling normal keyevent
	        iPreviousKeyeventNormal = EFalse;
	        iRetainKeyBoardtypeFornextkeyevent = EFalse;	        
	        }
	        
	        //Additional modifier set by query dialog, while simulating event
	        // As raw key events are used to simulate, modifiers and scan codes cannot be set at a time
	        // In the 1st event, modifiers are set and in subsequent event, scan codes will be set. 
	        if(aKeyEvent.iModifiers & EModifierSpecial)
	        	{
	        	iRetainKeyBoardtypeFornextkeyevent = ETrue;   
	        	iPreviousKeyeventSpecial = ETrue;
	        	iPreviousKeyeventNormal = ETrue;     	
	        	return EKeyWasConsumed;	        
	        	}	        							
		}
	else if(iFepMan.KeyboardLayout() == EPtiKeyboard12Key && iPreviousKeyeventSpecial)	 
		{
		//In case of query dialog, treat it as special event, till keyup event.
		if(!iRetainKeyBoardtypeFornextkeyevent)
			{		
			TInt activeKeypad=0;
			TInt activeKeypadType =0;
			RProperty::Get(KCRUidAvkon, KAknKeyBoardLayout,activeKeypad);
	        RProperty::Get(KCRUidAvkon, KAknQwertyInputModeActive, activeKeypadType);
			iFepMan.SetKeyboardLayout((TPtiKeyboardType)activeKeypad);
	        iFepMan.SetFnKeyMappingState();
	        iFepMan.SetQwertyMode(activeKeypadType);
	        iPreviousKeyeventNormal = ETrue;
	        iPreviousKeyeventSpecial = EFalse;
			}
		else if(aEventCode == EEventKeyUp)	//In case of query dialog reset flag in Keyup event.			
			iRetainKeyBoardtypeFornextkeyevent =EFalse;	
		}
		
	if(iPreviousKeyeventSpecial)
		{
		iFepMan.CancelShiftKeyTimer();
		iFepMan.ClearFlag(CAknFepManager::EFlagShiftKeyDepressed);					
		}
		
	return EKeyWasNotConsumed;
	}
	
// End of file