fep/aknfep/src/AknFepCaseManager.cpp
branchRCL_3
changeset 44 ecbabf52600f
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/fep/aknfep/src/AknFepCaseManager.cpp	Wed Sep 01 12:23:33 2010 +0100
@@ -0,0 +1,1062 @@
+/*
+* 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:           
+*
+*/
+
+
+
+
+
+
+
+
+
+
+
+
+#include "AknFepCaseManager.h"
+#include "AknFepPanic.h"
+#include "aknfeppeninputenums.h"
+
+#include <aknfep.rsg>
+#include <aknedsts.h>
+#include <barsread.h>
+#include <jplangutil.h>
+#include <AknPictographInterface.h>
+#include <AknPictographDrawerInterface.h>
+#include <featmgr.h>
+#include <PUAcodes.hrh>
+
+const TText KSpaceCharacter=' ';
+const TText KInvertedExclamationMarkCharacter = 0x00A1;
+const TText KInvertedQuestionMarkCharacter = 0x00BF;
+
+/* Teleca change begin, 18.08.2009 ssal */
+const TInt KSBase = 0xac00; // base address for hangul syllables
+const TInt KCBase = 0x3131; // base address for hangul compatibility jamo
+
+const TInt KLCount = 19; // count of L-jamos
+const TInt KVCount = 21; // count of V-jamos
+const TInt KTCount = 28; // count of T-jamos
+const TInt KNCount = KVCount * KTCount; // count of V&T-jamo combinations
+const TInt KSCount = KLCount * KNCount; // total count of hangul syllables
+
+// Conversion table for hangul compatibility jamo
+const TUint16 conversionTableFromCompatibilityJamoToJamo[] = {
+0x1100,
+0x1101,
+0x11AA,
+0x1102,
+0x11AC,
+0x11AD,
+0x1103,
+0x1104,
+0x1105,
+0x11B0,
+0x11B1,
+0x11B2,
+0x11B3,
+0x11B4,
+0x11B5,
+0x11B6,
+0x1106,
+0x1107,
+0x1108,
+0x11B9,
+0x1109,
+0x110A,
+0x110B,
+0x110C,
+0x110D,
+0x110E,
+0x110F,
+0x1110,
+0x1111,
+0x1112,
+0x1161,
+0x1162,
+0x1163,
+0x1164,
+0x1165,
+0x1166,
+0x1167,
+0x1168,
+0x1169,
+0x116A,
+0x116B,
+0x116C,
+0x116D,
+0x116E,
+0x116F,
+0x1170,
+0x1171,
+0x1172,
+0x1173,
+0x1174,
+0x1175,
+};
+
+#define KCCount (sizeof(conversionTableFromCompatibilityJamoToJamo) / sizeof(TUint16))
+/* Teleca change end, 18.08.2009 ssal */
+
+// 0x2029 is the paragraph seperator. When the user inputs a
+// new line character, 0x2029 is inserted.
+const TText KParagraphSeperator = 0x2029;
+// 0x000A is line feed character 
+const TText KLineFeed = 0x000A;
+
+//for chinese unicode ox4E00 to 0x9FA5
+const TText KChineseLowerUnicode = 0x4E00;
+const TText KChineseUpperUnicode = 0x9FA5;
+CAknFepCaseManager* CAknFepCaseManager::NewL(CAknFepManager& aFep)
+    {
+    CAknFepCaseManager* self = new (ELeave) CAknFepCaseManager(aFep);
+    CleanupStack::PushL(self);
+    self->ConstructL();
+    CleanupStack::Pop(); //self
+    return self;
+    }
+
+
+CAknFepCaseManager::~CAknFepCaseManager()
+    {
+    delete iPictoInterface;
+    }
+
+
+
+
+void CAknFepCaseManager::ConfigureCaseStateFromEditorState()
+    {
+    CAknEdwinState* editorState = iFep.EditorState();
+    if (!editorState)
+        {
+        return;
+        }
+        
+ 	iCurrentCase = editorState->CurrentCase();   
+    iPreviousCase = editorState->CaseState().PreviousCase();    
+        
+    SetFixedCase(editorState->Flags() & EAknEditorFlagFixedCase);
+
+    iPreviousCase = iCurrentCase;
+
+  	if (CurrentEditorIsFixedCase())
+    	{    
+    	SetCurrentCase(editorState->DefaultCase()); 
+    	iPreviousCase = iCurrentCase; 
+    	__ASSERT_DEBUG(IsCasePermitted(iCurrentCase), 
+                     AknFepPanic(EAknFepPanicNonPermittedEditorMode));		   	
+    	} 
+    	
+    //determine it based on position 
+    UpdateCase(ENullNaviEvent);
+
+    // if still there is no current case set - use the default
+    if (!iCurrentCase)
+        {
+        SetCurrentCase(editorState->DefaultCase());
+        if (!IsCasePermitted(iCurrentCase))
+            {
+            SetCurrentCase(editorState->CurrentCase());
+            }
+        
+        iPreviousCase = iCurrentCase;
+        __ASSERT_DEBUG(IsCasePermitted(iCurrentCase), 
+                       AknFepPanic(EAknFepPanicNonPermittedEditorMode));
+        }
+
+    // if the current is not allowed stay where we are
+    else if (!IsCasePermitted(iCurrentCase))
+        {
+        if (!iPreviousCase)
+            {
+            SetCurrentCase(editorState->DefaultCase());
+            iPreviousCase = iCurrentCase;
+            __ASSERT_DEBUG(IsCasePermitted(iCurrentCase), 
+                           AknFepPanic(EAknFepPanicNonPermittedEditorMode));
+            }
+        else
+            {
+            if (IsCasePermitted(iPreviousCase))
+                {
+                SetCurrentCase(iPreviousCase);
+                }
+            else
+                {
+                SetCurrentCase(editorState->DefaultCase());
+                iPreviousCase = iCurrentCase;
+                __ASSERT_DEBUG(IsCasePermitted(iCurrentCase), 
+                               AknFepPanic(EAknFepPanicNonPermittedEditorMode));
+                }    
+            }
+        }
+    }
+
+
+void CAknFepCaseManager::UpdateCase(TWesternNaviEvent aNaviEvent)
+    {
+    TBool applyCaseNeeded = ETrue;
+    if ( iFep.IsFlagSet(CAknFepManager::EFlagSupressAutoUpdate) )
+        {
+        return;
+        }
+    CAknEdwinState* editorState = iFep.EditorState();
+    if (!CurrentEditorIsFixedCase() && editorState 
+      && editorState->PermittedCases() & EAknEditorTextCase)
+        {
+        // lets figure out where the cursor will be when the case needs to be checked
+        TInt cursorPos = FindCursorPositionAtCaseCheck(aNaviEvent);
+        
+        if (!(cursorPos == 0 
+           && iFep.IsFlagSet(CAknFepManager::EFlagSupressAutoUpdateAtEditorStart) ))
+            {
+            TInt defaultCase = editorState->DefaultCase();
+            if ( cursorPos == 0 && 
+               (defaultCase == EAknEditorLowerCase || defaultCase == EAknEditorUpperCase ))
+                {
+                SetCurrentCase(defaultCase);
+                }
+            // UPPER CASE : If in upper case & inside sentence, stay in upper.
+            else if (!(iCurrentCase == EAknEditorUpperCase && 
+                       CapsLockRequired(cursorPos, iCurrentCase) && 
+                       aNaviEvent != EBackspaceEvent))
+                {
+                const TBool textCaseRequired = TextCaseRequired(cursorPos);
+                // UPPER CASE : If upper case & inside sentence & backspace event, stay in upper.
+                if (!(iCurrentCase == EAknEditorUpperCase && aNaviEvent == EBackspaceEvent &&
+                    !textCaseRequired && cursorPos != 0))
+                    {
+                    // TEXT CASE : set if at start of editor or following a full stop / text case character
+                    if (cursorPos == 0 || textCaseRequired)
+                        {
+                        TInt inputMode = iFep.InputMode();
+                        if (iFep.IsChineseInputLanguage() && 
+                        	inputMode != ELatin &&
+                        	iCurrentCase != EAknEditorLowerCase)
+                        {
+                        SetCurrentCase(iCurrentCase);
+                        }
+#ifdef RD_HINDI_PHONETIC_INPUT	
+                        
+	                    else if(iFep.IsIndicPhoneticInputLanguage())
+	                    	{
+	                    	if(CurrentCase() != EAknEditorUpperCase)
+	                    	    SetCurrentCase(EAknEditorLowerCase);	
+	                    	}
+#endif
+                        else
+                        {
+                        /* Teleca change begin, 27.08.2008 ssal 
+                        	original code:
+                        SetCurrentCase(EAknEditorTextCase);	*/
+                        
+                        if(iFep.IsKoreanInputLanguage() && inputMode == EHangul )
+                        	{
+                        	SetCurrentCase(EAknEditorLowerCase);	
+                        	}
+                        else
+                        	{
+                            SetCurrentCase(EAknEditorTextCase);
+                            }
+                        /* Teleca change end, 27.08.2008 ssal */
+                        }
+                        applyCaseNeeded = EFalse;
+                        }
+                    else // OTHERWISE LOWER
+                        {
+                        SetCurrentCase(EAknEditorLowerCase);
+                        applyCaseNeeded = EFalse;
+                        }
+                    }
+                }
+            }
+        else
+            {
+            iFep.ClearFlag(CAknFepManager::EFlagSupressAutoUpdateAtEditorStart);
+            }
+        }
+    if (applyCaseNeeded)
+        {
+        ApplyCase();
+        }
+    }
+
+
+void CAknFepCaseManager::ManualCaseUpdate()
+    {
+    if (!CurrentEditorIsFixedCase())
+        {
+        if (IsCasePermitted(iCurrentCase)) 
+            {
+            DoManualCaseUpdate();
+            iFep.ClearFlag(CAknFepManager::EFlagSupressAutoUpdate);
+            }
+        }
+    }
+
+
+void CAknFepCaseManager::RevertCaseChange()
+    {
+    if (!CurrentEditorIsFixedCase())
+        {
+        if (IsCasePermitted(iPreviousCase)) 
+            {
+            SetCurrentCase(iPreviousCase); 
+            }
+        }
+    }
+
+
+void CAknFepCaseManager::SetCurrentCase(TInt aCase, TBool aOverrideNumericTest)
+    {
+    if ( ( iFep.InputMode() == ENumber || iFep.InputMode() == ENativeNumber ) && 
+        !aOverrideNumericTest)
+        {
+        if ( iFep.EditorState() && 
+            (iFep.EditorState()->NumericKeymap() == EAknEditorConverterNumberModeKeymap) )
+            {
+            // Converter number mode keymap for '*' key requires upper case 'E'.
+            iCurrentCase = EAknEditorUpperCase;
+            }
+        else
+            {
+            // Number mode keymappings for '*' -key uses lower case letters (pw).
+            iCurrentCase = EAknEditorLowerCase;
+            }
+        }
+    else
+        {
+        if ( aCase == ECaseInverseText )
+            {
+            iCurrentCase = ECaseUpper;
+            iInverseCase = ETrue;
+            }
+        else
+            {
+            iCurrentCase = aCase;
+            iInverseCase = EFalse;
+            }
+        }
+    ApplyCase();
+    }
+
+
+TInt CAknFepCaseManager::CurrentCase() const
+    {
+    return iCurrentCase;
+    }
+
+TBool CAknFepCaseManager::IsAbleChangeCase() const
+    {
+    TBool isable = ETrue;
+    
+    CAknEdwinState* editorState = iFep.EditorState();
+    if (editorState)
+    	{
+	    const TInt permittedCases = editorState->PermittedCases();
+	    
+	    if (permittedCases == EAknEditorUpperCase ||
+	        permittedCases == EAknEditorLowerCase )
+	        {
+	        isable = EFalse;
+	        }
+	    else if ((editorState->Flags() & EAknEditorFlagForceTransparentFepModes)
+	          && iFep.IsChineseInputLanguage())
+	        {
+	        isable = EFalse;
+	        }
+	    else if ( editorState->Flags() & EAknEditorFlagFixedCase )
+	        {
+	        isable = EFalse;
+	        }
+	    else if (!iFep.InputLanguageSupportsCaseChanges())
+	        {
+	        isable = EFalse;
+	        }
+    	}
+    else
+    	{
+    	isable = EFalse;
+    	}
+    return isable;
+    }
+
+CAknFepCaseManager::CAknFepCaseManager(CAknFepManager& aFep) :
+    iFep(aFep)
+    {
+    }
+
+
+void CAknFepCaseManager::ConstructL()
+    {
+    // Set text case character set from resource
+    TResourceReader reader;
+    CCoeEnv::Static()->CreateResourceReaderLC(reader, R_AKNFEP_TEXT_CASE_CHARACTER_SET);
+    iTextCaseCharacters=reader.ReadTPtrC();
+
+    // Create PictographInterface
+    CCoeControl* control = NULL;
+    MAknPictographAnimatorCallBack* callback = NULL;
+    iPictoInterface = CAknPictographInterface::NewL(*control,*callback);
+
+    CleanupStack::PopAndDestroy(); // reader*/
+    }
+
+
+TInt CAknFepCaseManager::FindCursorPositionAtCaseCheck(TWesternNaviEvent aNaviEvent)
+    {
+    TCursorSelection curCursSel;
+    iFep.FepAwareTextEditor()->GetCursorSelectionForFep(curCursSel);
+    
+    TInt docLength = iFep.FepAwareTextEditor()->DocumentLengthForFep();
+    TInt returnedPosition = -1;
+
+    TCursorSelection uncommittedText = iFep.UncommittedText();
+
+    switch (aNaviEvent)
+        {
+        case ENullNaviEvent:
+            if (iFep.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction))
+                {
+                returnedPosition = uncommittedText.HigherPos();
+                }
+            else
+                {
+                // In case of non-predictive mode, following condition
+                // check text is selceted mode or not?
+                // In selected mode, character always being inserted
+                // anchor position
+                if( curCursSel.Length() >0 )
+                    {
+                    returnedPosition = curCursSel.LowerPos();
+                    }
+                else
+                    {
+                    returnedPosition = curCursSel.HigherPos();
+                    }
+                }
+            break;
+        case ERightNaviEvent:
+            if (iFep.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction))
+                {
+                returnedPosition = uncommittedText.HigherPos();
+                }
+            else
+                {
+                if (curCursSel.Length())
+                    {
+                    returnedPosition = curCursSel.HigherPos();
+                    }
+                else
+                    {
+                    if (curCursSel.iCursorPos == docLength)
+                        {
+                        returnedPosition = 0;
+                        }
+                    else
+                        {
+                        returnedPosition = curCursSel.HigherPos() + 1;
+                        }
+                    }
+                }
+            break;
+        case ELeftNaviEvent:
+            if (iFep.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction))
+                {
+                returnedPosition = uncommittedText.LowerPos();
+                }
+            else
+                {
+                if (curCursSel.Length())
+                    {
+                    returnedPosition = curCursSel.LowerPos();
+                    }
+                else
+                    {
+                    if (curCursSel.iCursorPos == 0)
+                        {
+                        returnedPosition = docLength;
+                        }
+                    else
+                        {
+                        returnedPosition = curCursSel.LowerPos() - 1;
+                        }
+                    }
+                }
+            break;
+        case EBackspaceEvent:
+            if (iFep.IsFlagSet(CAknFepManager::EFlagInsideInlineEditingTransaction))
+                {
+                returnedPosition = uncommittedText.HigherPos();
+                }
+            else
+                {
+                if (curCursSel.Length())
+                    {
+                    returnedPosition = curCursSel.LowerPos();
+                    }
+                else
+                    {
+                    returnedPosition = curCursSel.LowerPos() - 1;
+                    }
+                }
+            break;
+        default:
+            break;
+        }
+
+    if (returnedPosition < 0)
+        {
+        returnedPosition = 0;
+        }
+    else if (returnedPosition > docLength)
+        {
+        returnedPosition = docLength;
+        }
+    return returnedPosition;
+    }
+
+
+TBool CAknFepCaseManager::TextCaseRequired(TInt aCursorPos)
+    {
+    TBool textCaseRequired = EFalse;
+    TInt charPosToCheck = aCursorPos - 1;   
+    TChar ch = KSpaceCharacter;
+    TBuf<CAknFepManager::ESingleCharacter> editorText;
+    TInt nonEnterSpaceCharCount = 0;
+    
+    if (charPosToCheck >= 0) 
+        {
+        // Get the character just before the cursor.
+        iFep.FepAwareTextEditor()->GetEditorContentForFep(editorText,
+            charPosToCheck, CAknFepManager::ESingleCharacter);
+        ch = editorText[0];        
+        if ( ch == KInvertedQuestionMarkCharacter || ch == KInvertedExclamationMarkCharacter )
+            {
+            // Text case is required imediatelly after 'spanich' question and exclamation mark that begins a sentence.
+            textCaseRequired = ETrue;        
+            }
+        // for chinese uncode limit
+        if ( ch >= KChineseLowerUnicode && ch<= KChineseUpperUnicode)
+        	{
+        	textCaseRequired = ETrue; 
+        	}
+        
+        if( ( TUint )ch == KParagraphSeperator || ( TUint )ch == KLineFeed )
+            {
+            // When we have enter key, we need to decide the case after: 
+            // we reach some non space characters (OR)
+            // we reach the beginning of the editor
+            while (charPosToCheck > 0 && !textCaseRequired)
+                {
+                charPosToCheck--;
+                iFep.FepAwareTextEditor()->GetEditorContentForFep(editorText,
+                    charPosToCheck, CAknFepManager::ESingleCharacter);
+                ch = editorText[0];
+                
+                // IsSpace returns true for ParagraphSeperator.
+                if (!(ch.IsSpace() || ch == EKeyEnter) && nonEnterSpaceCharCount <= 1)
+                    {
+                    if (IsTextCaseCharacter(ch))
+                        {
+        	            textCaseRequired = ETrue;
+                        }
+                    break;
+                    }
+                else 
+                	{
+                	if(!( ( TUint )ch == KParagraphSeperator || ( TUint )ch == EKeyEnter || ( TUint )ch == KLineFeed ) )
+                		{
+                		// Incrementing the count if we have
+                		// IsSpace() returns True AND
+                		// the character is not a ParagraphSeparator or Enter or Line Feed
+                		nonEnterSpaceCharCount++;
+	                	}
+                	if (charPosToCheck <= 0 && !nonEnterSpaceCharCount)
+                    	{
+                    	//Reached the beginning of the Editor. Text case should be ON
+                    	textCaseRequired = ETrue;
+                    	}                    	
+                	}    	
+                }
+            charPosToCheck = aCursorPos - 1;
+            
+           	if (charPosToCheck <= 0 && !nonEnterSpaceCharCount)
+           	    {
+           	    // If Enter is pressed as the first character. Text case is ON
+                textCaseRequired = ETrue;
+                }                    	
+            
+        	}
+
+        /* Teleca change begin, 27.08.2008 ssal */	
+        if (IsKoreanCharacter(ch))
+        	{
+        	textCaseRequired = ETrue;       
+        	}
+        /* Teleca change end, 27.08.2008 ssal */
+        }
+
+
+    if (charPosToCheck > 0 && !textCaseRequired) 
+        {
+        if (iFep.IsFeatureSupportedJapanese())
+            {
+            // The character case is changed to text case when the left side
+            // character is Japanese character.
+            if (IsJapaneseCharacter(ch))
+                {
+                textCaseRequired = ETrue;
+                }
+            else if (ch.IsSpace() || ch == EKeyEnter ||
+                     ch == (TUint)KPuaCodeSpaceSymbol)
+                {
+                while (charPosToCheck > 0)
+                    {
+                    charPosToCheck--;
+                    // Get the character before the previous character.
+                    iFep.FepAwareTextEditor()->GetEditorContentForFep(editorText,
+                        charPosToCheck, CAknFepManager::ESingleCharacter);
+                    ch = editorText[0];
+                    if (!(ch.IsSpace() || ch == EKeyEnter || ch == (TUint)KPuaCodeSpaceSymbol))
+                        {
+                        if (IsTextCaseCharacter(ch))
+                            {
+                            // One space between the cursor and the punctuation character => text case needed
+                            textCaseRequired = ETrue;
+                            }
+                        break;
+                        }
+                    else if (charPosToCheck <= 0)
+                        {
+                        // reached the start of the editor => text case needed
+                        textCaseRequired = ETrue;
+                        }
+                    }
+                }
+            }
+        else
+            {
+            if ( charPosToCheck > 0 && !textCaseRequired && !nonEnterSpaceCharCount) 
+			    {        
+			    if ( ch.IsSpace() || ch == EKeyEnter || 
+			    	ch == ( TUint )KPuaCodeSpaceSymbol )
+			        {
+			        while ( charPosToCheck > 0 )
+			            {
+			            charPosToCheck--;
+			            
+			            // Get the character before the previous character.
+			            iFep.FepAwareTextEditor()->GetEditorContentForFep(editorText,
+			                charPosToCheck, CAknFepManager::ESingleCharacter);
+			                
+			            ch = editorText[0];
+			            
+			            // This condition is added to keep Text case ON when space is
+			            // entered after a "New line" character.
+			            if( ( TUint )ch == KParagraphSeperator )
+			            	{			            	
+			            	textCaseRequired = ETrue;
+			            	break;
+			            	}
+			            	
+			            if ( !( ch == EKeyEnter || 
+			            		ch == ( TUint )KPuaCodeSpaceSymbol ) )
+			                {
+			                if ( IsTextCaseCharacter( ch ) )
+			                    {			                    
+			                    textCaseRequired = ETrue;
+			                    }
+			                break;
+			                }
+			            else if ( charPosToCheck <= 0 )
+			                {
+			                // reached the start of the editor => text case needed
+			                textCaseRequired = ETrue;
+			                }
+			            }
+                    }
+                }
+            }
+        }
+    else if (charPosToCheck < 0)
+        {
+        // reached the start of the editor => text case needed
+        textCaseRequired = ETrue;
+        }
+    else    // if (charPosToCheck == 0)
+        {
+        if (!textCaseRequired && iFep.IsFeatureSupportedJapanese())
+            {
+            if (IsJapaneseCharacter(ch)
+             || ch.IsSpace()
+             || ch == EKeyEnter
+             || ch == (TUint)KPuaCodeSpaceSymbol)
+                {
+                // reached the start of the editor => text case needed
+                textCaseRequired = ETrue;
+                }
+            }
+        }
+
+    return textCaseRequired;
+    }
+
+
+TBool CAknFepCaseManager::CapsLockRequired(TInt aCursorPos, TInt aCurCase)
+    {
+    TBool capsLocked = iForceUpperCase;
+    TInt charPosToCheck = aCursorPos - 1;   
+    TChar ch = KSpaceCharacter;
+    TBuf<CAknFepManager::ESingleCharacter> editorText;
+    
+    // is the previous character alphabetical?
+    if (charPosToCheck >= 0) 
+        {
+        iFep.FepAwareTextEditor()->GetEditorContentForFep(editorText,
+            charPosToCheck, CAknFepManager::ESingleCharacter);
+        if( editorText.Length()>0 )
+            {
+            ch = editorText[0];
+            }
+            
+        if (IsAlpha(ch) ||
+            ((aCurCase == EAknEditorUpperCase || aCurCase == ECaseInverseText) && 
+             !TextCaseRequired(aCursorPos)))
+            {
+            // Caps lock (upper case mode) is kept up if we are currently in upper case mode
+            // and previous character is alpha character or for example full stop. 
+            // If the cursor and full stop is separated with space, text case is restored. 
+            capsLocked = ETrue;
+            }
+        }
+    
+    return capsLocked;
+    }
+
+
+void CAknFepCaseManager::DoManualCaseUpdate()
+    {
+    TBool applyCaseNeeded = ETrue;
+    iPreviousCase = iCurrentCase;
+    TCursorSelection currentSelection;
+    iFep.FepAwareTextEditor()->GetCursorSelectionForFep(currentSelection);
+
+    switch (iCurrentCase)
+        {
+        case EAknEditorLowerCase:
+            if (CapsLockRequired(currentSelection.iCursorPos, iCurrentCase) 
+             && IsCasePermitted(EAknEditorUpperCase))
+                {
+                SetCurrentCase(EAknEditorUpperCase);
+                }
+            else if (!IsCasePermitted(EAknEditorTextCase) 
+#ifdef RD_HINDI_PHONETIC_INPUT	
+            
+            || iFep.IsIndicPhoneticInputLanguage()
+#endif            
+            )
+                {
+                if( IsCasePermitted(EAknEditorUpperCase) )
+                    SetCurrentCase(EAknEditorUpperCase);
+                }
+            else
+                {
+                SetCurrentCase(EAknEditorTextCase);
+                }
+            applyCaseNeeded = EFalse;
+            break;
+        case EAknEditorTextCase:
+             if (iForceUpperCase && IsCasePermitted(EAknEditorUpperCase))
+                {
+                SetCurrentCase(EAknEditorUpperCase);
+                applyCaseNeeded = EFalse;     
+                break;          
+                }
+        case EAknEditorUpperCase:
+            SetCurrentCase(EAknEditorLowerCase);
+            applyCaseNeeded = EFalse;
+            break;
+        default:
+            break;
+        }
+
+    if (!(IsCasePermitted(iCurrentCase)))
+        {
+        DoManualCaseUpdate();
+        }
+    else
+        {
+        if (applyCaseNeeded)
+            {
+            ApplyCase();
+            }
+        }
+    }
+
+
+void CAknFepCaseManager::ApplyCase()
+    {
+    // update engine...
+    switch (iCurrentCase)
+        {
+        case EAknEditorUpperCase: // fall through
+        case EAknEditorTextCase:
+            iFep.SetCase(EUpperCase);
+            break;
+        case EAknEditorLowerCase: // fall through
+        default:
+            iFep.SetCase(ELowerCase);
+            break;
+        }
+    if(iFep.EditorState())
+        {
+        iFep.EditorState()->SetCurrentCase(iCurrentCase);
+        }
+    iFep.UpdateIndicators();
+
+#ifdef RD_SCALABLE_UI_V2    
+    iFep.UpdateTouchCaseMode();
+#endif
+    }
+
+
+void CAknFepCaseManager::SetFixedCase(TBool aFixedCase)
+    {
+    iFixedCase = aFixedCase;
+    }
+
+
+TBool CAknFepCaseManager::CurrentEditorIsFixedCase() const
+    {
+    return iFixedCase;
+    }
+
+
+TBool CAknFepCaseManager::IsCasePermitted(TInt aCase) const
+    {
+    // if current is permitted or current is lower and text is permitted
+    CAknEdwinState* editorState = iFep.EditorState();
+    return ( editorState && ( !editorState->PermittedCases() || 
+           (aCase & editorState->PermittedCases() || 
+           (aCase == EAknEditorLowerCase && 
+           (EAknEditorTextCase & editorState->PermittedCases())))) );
+    }
+
+TBool CAknFepCaseManager::IsTextCaseCharacter(TChar aChar) const
+    {
+    TBool ret = EFalse;
+    if (iTextCaseCharacters.Locate(aChar)!=KErrNotFound)
+        {
+        ret = ETrue;
+        }
+    else if (JPLangUtil::IsKanji((TText)aChar)
+          || JPLangUtil::IsHiragana((TText)aChar)
+          || JPLangUtil::IsKatakana((TText) aChar))
+        {
+        ret = ETrue;
+        }
+    return ret;
+    }
+
+TBool CAknFepCaseManager::IsAlpha(TChar aChar) const
+    {
+    TBool ret = EFalse;
+    TBool isPicto = EFalse;
+    
+    if (iPictoInterface && iPictoInterface->Interface())
+        {
+        isPicto = iPictoInterface->Interface()->IsPictograph((TText)aChar);
+        }
+    
+    if (!( JPLangUtil::IsHiragana((TText)aChar)
+        || JPLangUtil::IsKanji((TText)aChar)
+        || JPLangUtil::IsKatakana((TText)aChar)
+        /* Teleca change begin, 18.08.2009 ssal */
+        || IsKoreanCharacter(aChar)
+        /* Teleca change end, 18.08.2009 ssal */
+        || isPicto ))
+        {
+        TChar ch = aChar;
+        if (iFep.IsFeatureSupportedJapanese() && 
+            JPLangUtil::IsFullWidth((TText) ch))
+            {
+            TBuf<CAknFepManager::ESingleCharacter> fullChar;
+            TBuf<CAknFepManager::ESingleCharacter> halfChar;
+            fullChar.Append(ch);
+            if (JPLangUtil::ConvertFullToHalfWidth(fullChar, halfChar))
+                {
+                ch = halfChar[0];
+                ret = ch.IsAlpha();
+                }
+            }
+        else
+            {
+            ret = ch.IsAlpha();
+            }
+        }
+    return ret;
+    }
+
+TInt CAknFepCaseManager::PluginPermittedCases()
+    {
+	TInt allowedCases = ECaseInvalide;    
+    if ( !(iFep.EditorState()) )
+        {
+        return allowedCases;
+        }
+    
+    TInt editorCases = iFep.EditorState()->PermittedCases();
+    
+    if( editorCases == 0 )
+        {
+        allowedCases = ECaseUpper | ECaseLower | ECaseText | ECaseInverseText;    
+        }
+    else
+        {
+        allowedCases = editorCases;
+        
+        if( (editorCases & EAknEditorUpperCase) && 
+            (editorCases & EAknEditorTextCase) )
+            {
+            allowedCases |= ECaseInverseText;
+            }
+        }
+    
+    return allowedCases;
+    }
+
+
+// -----------------------------------------------------------------------------
+// CAknFepCaseManager::CapsLockRequired
+// 
+// 
+// -----------------------------------------------------------------------------
+//
+TBool CAknFepCaseManager::CapsLockRequired(TInt aCurrentCase)
+    {
+  	TCursorSelection currentSelection;
+  	
+  	if (iFep.FepAwareTextEditor())
+  		{  		  	
+    	iFep.FepAwareTextEditor()->GetCursorSelectionForFep(currentSelection);    
+    	return CapsLockRequired(currentSelection.iCursorPos, aCurrentCase);
+  		}
+  		
+	return EFalse;  		
+    }
+
+// -----------------------------------------------------------------------------
+// CAknFepCaseManager::IsJapaneseCharacter
+// Determines whether a character is one of Japanese characters
+// -----------------------------------------------------------------------------
+//
+TBool CAknFepCaseManager::IsJapaneseCharacter(TChar aChar) const
+    {
+    TBool ret = EFalse;
+    // The character case is changed to text case when the left side
+    // character is Japanese character (Hiragana, Katakana, Kanji,
+    // Full-width Symbol).
+    if (JPLangUtil::IsKanji((TText)aChar)
+     || JPLangUtil::IsHiragana((TText)aChar)
+     || JPLangUtil::IsKatakana((TText)aChar))
+        {
+        // Hiragana, Katakana, Kanji
+        ret = ETrue;
+        }
+    else if (JPLangUtil::IsFullWidth((TText)aChar)
+     && aChar != KPuaCodeSpaceSymbol
+     && aChar != KPuaCodeLineFeedSymbol)
+        {
+        if (iFep.InputMode() == ELatin
+          && iFep.CharacterWidth() == EFullWidthChar)
+            {
+            TBuf<CAknFepManager::ESingleCharacter> fullChar;
+            TBuf<CAknFepManager::ESingleCharacter> halfChar;
+            fullChar.Append(aChar);
+            if (JPLangUtil::ConvertFullToHalfWidth(fullChar, halfChar))
+                {
+                TChar ch2 = halfChar[0];
+                if (!ch2.IsAlpha())
+                    {
+                    // Full width Symbol or Number
+                    ret = ETrue;
+                    }
+                }
+            else
+                {
+                // The character which isn't converted to half is
+                // special Japanese Symbols.
+                ret = ETrue;
+                }
+            }
+        else
+            {
+            ret = ETrue;
+            }
+        }
+    return ret;
+    }
+
+/* Teleca change begin, 18.08.2009 ssal */
+// -----------------------------------------------------------------------------
+// CAknFepCaseManager::IsKoreanCharacter
+// Determines whether a character is one of Korean characters
+// -----------------------------------------------------------------------------
+//
+TBool CAknFepCaseManager::IsKoreanCharacter(TChar aChar) const
+    {  
+    if ( IsHangulSyllable( aChar) || IsHangulCompatibilityJamo( aChar ) )
+    	{
+    	return ETrue;
+    	}
+    return EFalse;
+    }
+
+// ---------------------------------------------------------------------------
+// Checks that is given character hangul syllable or not.
+// ---------------------------------------------------------------------------
+//
+TBool CAknFepCaseManager::IsHangulSyllable( TChar aChar ) const
+    {
+    // Character is 'hangul syllable' 
+    // if it's numeric value is between KSBase and KSBase + KSCount.
+    TInt index = static_cast<TInt> ( aChar ) - KSBase;
+    if ( index < 0 || index >= KSCount )
+        {
+        return EFalse;
+        }
+    return ETrue;
+    }
+
+// ---------------------------------------------------------------------------
+// Checks that is given character hangul syllable or not.
+// ---------------------------------------------------------------------------
+//
+TBool CAknFepCaseManager::IsHangulCompatibilityJamo( TChar aChar ) const
+    {
+    // Character is 'hangul compatibility jamo' 
+    // if it's numeric value is between KCBase and KCBase + KCCount.
+    TInt index = static_cast<TInt> ( aChar ) - KCBase;
+    if ( index < 0 || index >= KCCount )
+        {
+        return EFalse;
+        }
+    return ETrue;
+    }
+
+/* Teleca change end, 18.08.2009 ssal */    
+// End of file