textinput/ptienginev2/src/PtiDefaultCore.cpp
author Pat Downey <patd@symbian.org>
Wed, 01 Sep 2010 12:23:33 +0100
branchRCL_3
changeset 44 ecbabf52600f
parent 43 ebd48d2de13c
permissions -rw-r--r--
Revert incorrect RCL_3 drop: Revision: 201033 Kit: 201035

/*
* Copyright (c) 2003-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:   Default core for predective text input engine.
*               Implements basic qwerty & multitapping input.  
*
*/


#include <e32property.h>
#include "PtiDefaultCore.h"
#include "PtiLanguage.h"
#include "PtiEngine.h"
#include "PtiKeyMappings.h"
#include <ptiengine.rsg>
#include "PtiDefaultNumberModeMappingdata.h"

#include <data_caging_path_literals.hrh> 

// CONSTANTS
const TInt KMultiTapTimeoutMS = 1000000;
//const TInt KPtiKeymappingsDefaultImplementation = 0x10281897;

_LIT(KDefaultCoreVendor, "Default core");

#ifdef _DEBUG
_LIT(KPtiEnginePanic, "CPtiDefaultCore");
#endif

// lam with alef ligatures are entered as one press but are actually two symbols
// These four arabic characters are handled as a special case.
const TUint16 KArarbicLigatureLamWithAlefWithMaddaAboveIsolatedForm = 0xFEF5;
const TUint16 KArarbicLigatureLamWithAlefWithHamzaAboveIsolatedForm = 0xFEF7;
const TUint16 KArarbicLigatureLamWithAlefWithHamzaBelowIsolatedForm = 0xFEF9;
const TUint16 KArarbicLigatureLamWithAlefIsolatedForm               = 0xFEFB;

_LIT(KArarbicLigatureLamWithAlefWithMaddaAboveIsolatedSequence, "\x0644\x622");
_LIT(KArarbicLigatureLamWithAlefWithHamzaAboveIsolatedSequence, "\x0644\x623");
_LIT(KArarbicLigatureLamWithAlefWithHamzaBelowIsolatedSequence, "\x0644\x625");
_LIT(KArarbicLigatureLamWithAlefIsolatedSequence, "\x0644\x627");


//
// CMultiTapTimer
//

// ---------------------------------------------------------------------------
// CMultiTapTimer::NewL
// 
// ---------------------------------------------------------------------------
// 
CMultiTapTimer* CMultiTapTimer::NewL(TCallBack aCallback)
	{
	CMultiTapTimer* self = new(ELeave)CMultiTapTimer(aCallback);
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop();	// self
	return self;
	}


// ---------------------------------------------------------------------------
// CMultiTapTimer::CMultitapTimer
// 
// ---------------------------------------------------------------------------
// 
CMultiTapTimer::CMultiTapTimer(TCallBack aCallback)
:CTimer(EPriorityStandard), iCallback(aCallback)
	{
	}

// ---------------------------------------------------------------------------
// CMultiTapTimer::ConstructL
// 
// ---------------------------------------------------------------------------
// 
void CMultiTapTimer::ConstructL()
	{
	CTimer::ConstructL();
	CActiveScheduler::Add(this);
	}

// ---------------------------------------------------------------------------
// CMultiTapTimer::After
// 
// ---------------------------------------------------------------------------
// 
void CMultiTapTimer::After(TTimeIntervalMicroSeconds32 aInterval)
	{
	if (IsActive())
		{
		Cancel();
		}
	CTimer::After(aInterval);
	}

// ---------------------------------------------------------------------------
// CMultiTapTimer::RunL
// 
// ---------------------------------------------------------------------------
// 
void CMultiTapTimer::RunL()
	{
	iCallback.CallBack();
	}


//
// CPtiDefaultCore
//

// ---------------------------------------------------------------------------
// CPtiDefaultCore::NewL
// 
// ---------------------------------------------------------------------------
// 
CPtiDefaultCore* CPtiDefaultCore::NewL(CPtiEngine* aOwner, TDes* aTextBuffer)
	{
	CPtiDefaultCore *core = new (ELeave) CPtiDefaultCore(aOwner, aTextBuffer);
	CleanupStack::PushL( core );
	core->ConstructL();
	CleanupStack::Pop(); // core
	return core;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::CPtiDefaultCore
// .
// ---------------------------------------------------------------------------
// 
CPtiDefaultCore::CPtiDefaultCore()
	{
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::CPtiDefaultCore
// 
// ---------------------------------------------------------------------------
// 
CPtiDefaultCore::CPtiDefaultCore(CPtiEngine* aOwner, TDes* aTextBuffer) : iOwner(aOwner), iTextBuffer(aTextBuffer)
	{
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::~CPtiDefaultCore
// 
// ---------------------------------------------------------------------------
// 
CPtiDefaultCore::~CPtiDefaultCore()
	{
	if (iTimer)
		{
		iTimer->Cancel();
		}
		 
    ShutDownKeymapFactory();
    					
	delete iTimer;	
	delete iNumericMappings;
	delete iNumericData;
	    
    iFactoryBindings.Close();    	
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::ConstructL
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::ConstructL()
	{
	FillCoreInfo();
	TCallBack timerExpire(TimerExpire, this);
	iTimer = CMultiTapTimer::NewL(timerExpire);
	iNumericData = CPtiNumberModeDataImpl::NewL();
	iNumericMappings = CPtiKeyMappings::NewL(iNumericData);  
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::InitializeL
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::InitializeL(CPtiEngine* aOwner, TDes* aTextBuffer, CArrayPtrFlat<CPtiCoreLanguage>* aLanguageList, TBool /*aUseDefautUserDictionary*/)
	{
	iOwner = aOwner;
	iTextBuffer = aTextBuffer;
    iLanguageList = aLanguageList;	
	
    TInt count = 0;
    TInt err = RProperty::Get(KPtiEnginePSUid, EKeyMapPropertyCount, count);
    if (err == KErrNone && count > 0)
        {
        // PS key found, keymap implementation list has been created and can be read
        TBuf16<256> dataBuffer;
        err = RProperty::Get(KPtiEnginePSUid, EKeyMapPropertyData, dataBuffer);
                                             
        if( err == KErrNone )
            {
            TInt langId;
            TInt factoryUid;
            for(TInt i = 0; i < count; i++ )
                {
                langId = dataBuffer[3*i];
                factoryUid = (dataBuffer[3*i+1] << 16) + dataBuffer[3*i+2];
            
                CPtiCoreLanguage* language = CPtiCoreLanguage::NewL();
                language->SetLanguageCode(langId);
		        language->SetCore(this, EPtiEngineNumeric);	

	            CleanupStack::PushL( language );     
	       	    aLanguageList->AppendL( language );
	            CleanupStack::Pop(); // language
             
                TPtiDataFactoryBinding newBind(langId, factoryUid);
                iFactoryBindings.AppendL(newBind);
                }
            return;
            }
        }

    // PS key not defined or error, we need to fetch the data from dll:s
	RArray<TInt> dataImpl;
	CleanupClosePushL(dataImpl);
	CPtiKeyMapDataFactory::ListImplementationsL(dataImpl);
   
	for (TInt i = 0; i < dataImpl.Count(); i++)
	    {
        ShutDownKeymapFactory();
	    iKeyMapDataFactory = CPtiKeyMapDataFactory::CreateImplementationL(TUid::Uid(dataImpl[i]));	  	    	  	   
	    ListDataFactoryLanguagesL(iKeyMapDataFactory, dataImpl[i], aLanguageList);	     
	    }
		
	if (!iFactoryBindings.Count())
	    {
	    // Cannot function without keymapping data.
	    User::Leave(KErrCorrupt);
	    } 
                     
    // Write language->dll mapping data to P/S-key so we don't need to load every dll later
    TBuf16<256> dataBuffer;
    for(TInt i = 0; i < iFactoryBindings.Count(); i++)
       {
       dataBuffer.Append(iFactoryBindings[i].iLanguage); // Language id should fit in 16 bits
       dataBuffer.Append(iFactoryBindings[i].iFactoryImplUid >> 16); // 32-bit Uid is appended in two parts
       dataBuffer.Append(iFactoryBindings[i].iFactoryImplUid);
       }
    
    // If these fail, then they just fail.                  
    RProperty::Set(KPtiEnginePSUid, EKeyMapPropertyCount, iFactoryBindings.Count());  
    RProperty::Set(KPtiEnginePSUid, EKeyMapPropertyData, dataBuffer);        
                  
	CleanupStack::PopAndDestroy(); // dataImpl    
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::OpenLanguageL
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::OpenLanguageL(CPtiCoreLanguage* aLanguage)
	{		
	TRAPD(err, LoadFactoryForLanguageL(aLanguage->LanguageCode()));
	
	ClearFlag(EPtiFlagVowelSeq);	

	return err;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::CloseLanguageL
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::CloseLanguageL()
	{
	if (iOwner->CurrentLanguage())
	    {
	    CPtiCoreLanguage* curLang = static_cast<CPtiCoreLanguage*>(iOwner->CurrentLanguage());
	    curLang->SetKeyMapDataFactory(NULL);  
	    }			
	
	ClearFlag(EPtiFlagVowelSeq);	
	return KErrNone;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::GetCurrentWord
// 
// ---------------------------------------------------------------------------
// 
TPtrC CPtiDefaultCore::GetCurrentWord() const
	{
	__ASSERT_DEBUG(iTextBuffer, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	return TPtrC(iTextBuffer->Ptr());
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::CommitCurrentWord
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::CommitCurrentWord()
	{
	ClearBuffer();
	return KErrNone;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::NumberOfCandidates
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::NumberOfCandidates()
	{
	__ASSERT_DEBUG(iTextBuffer, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	if (iTextBuffer->Length() == 0)
		{
		return 0;	
		}

	return 1;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::FillCoreInfo
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::FillCoreInfo()
	{
	iCoreInfo.SetCapsBits(0);
	iCoreInfo.SetMaxWordLength(KDefaulCoreMaximumWordLength);
	iCoreInfo.SetVendorString( KDefaultCoreVendor );
	iCoreInfo.SetMaxNumberOfCandidates(1);
	iCoreInfo.SetUid(KDefaultCoreUid);
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::SetReordering
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::SetReordering(TBool aStatus)
	{
	if (aStatus)
		{
		return KErrNotSupported;
		}

	return KErrNone;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::AppendKeyPress
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::AppendKeyPress(TPtiKey aKey)
	{
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));

	CPtiCoreLanguage* lang = static_cast<CPtiCoreLanguage*>(iOwner->CurrentLanguage());
	const TPtiEngineInputMode inputMode = iOwner->InputMode();
	const TBool qwerty = iOwner->IsQwertyBasedMode(inputMode);
	TBool append = ETrue;
	TUint16 overridingDeadKey = 0;
	
	ClearFlag(EPtiGetAllFlag);		
	
	__ASSERT_DEBUG(lang, User::Panic(KPtiEnginePanic, KErrCorrupt));

	MPtiKeyMappings* maps = NULL;

	if (inputMode == EPtiEngineNumeric)
		{
		maps = iNumericMappings;
		}
    else if (inputMode == EPtiEngineHalfQwerty)		
        {
        if (lang->HasInputMode(EPtiEngineHalfQwerty))
            {
	        maps = lang->GetHalfQwertyKeymappings();            
            }        
        }		
	else if (qwerty)
		{
		if (lang->HasInputMode(EPtiEngineQwerty))
			{
			maps = lang->GetQwertyKeymappings();
			}
		}
	else
		{
		if (lang->HasInputMode(EPtiEngineMultitapping))
			{
			maps = lang->GetKeymappings();
			}
		}

	if (!maps)
		{
		iTimer->Cancel();
		return;
		}

	if (iTimer->IsActive())
		{
		TUint16 chr = maps->NextKey(aKey, append, iOwner->Case());
		if (chr && (!append) && (iTextBuffer->Length() > 0))
			{
			iTextBuffer->SetLength(iTextBuffer->Length() - 1);	
			}
		if (iTextBuffer->Length() < iTextBuffer->MaxLength())
			{
			if (chr)
				{
				iTextBuffer->Append(chr);
				}
			}
		}
	else
		{
		if (iTextBuffer->Length() < iTextBuffer->MaxLength())
			{
			if (qwerty)
				{
				CPtiQwertyKeyMappings* qmap = static_cast<CPtiQwertyKeyMappings*>(maps);		
				if (lang->LanguageCode() == ELangVietnamese)
					{
					qmap->SetFlag(CPtiQwertyKeyMappings::ETrackVietnameseToneMarksAndVowels);
					}
				else
					{
					qmap->ResetFlag(CPtiQwertyKeyMappings::ETrackVietnameseToneMarksAndVowels);
					}					
										
				if (qmap->DeadKey() && (aKey == qmap->CurrentKey()))
					{
					overridingDeadKey = qmap->DeadKeyRootChar();				
					} 
				}
			
			TUint16 chr = maps->StartMapping(aKey, iOwner->Case(), inputMode);
			
			if (chr == KPtiGetAllMarker) 
				{
				if (lang->LanguageCode() == ELangHindi)
					{									
					TPtrC ptr = maps->GetAll(iOwner->Case());				
					iTextBuffer->Append(ptr.Left(iTextBuffer->MaxLength() - iTextBuffer->Length()));
					SetFlag(EPtiGetAllFlag);		
					}
				}			
			else if (chr)
				{
                switch (chr)
                	{
          			case KArarbicLigatureLamWithAlefWithMaddaAboveIsolatedForm:
                 	     iTextBuffer->Append(KArarbicLigatureLamWithAlefWithMaddaAboveIsolatedSequence);
                 	     SetFlag(EPtiDoubleWideEntry);
                 	     break;
            		case KArarbicLigatureLamWithAlefWithHamzaAboveIsolatedForm:
                 	     iTextBuffer->Append(KArarbicLigatureLamWithAlefWithHamzaAboveIsolatedSequence);
                 	     SetFlag(EPtiDoubleWideEntry);                 
                 	     break;
            		case KArarbicLigatureLamWithAlefWithHamzaBelowIsolatedForm:
                 	     iTextBuffer->Append(KArarbicLigatureLamWithAlefWithHamzaBelowIsolatedSequence);
                         SetFlag(EPtiDoubleWideEntry);                 
                         break; 
                    case KArarbicLigatureLamWithAlefIsolatedForm:
                         iTextBuffer->Append(KArarbicLigatureLamWithAlefIsolatedSequence);      
                         SetFlag(EPtiDoubleWideEntry);                 
                         break;
                    default:       
                         iTextBuffer->Append(chr);    
                     }                
				}
				
			if (qwerty)
				{
				// There was dead key press waiting, but last key press didn't
				// contain dead key data, so both keys will be inserted.	
				CPtiQwertyKeyMappings* qmap = static_cast<CPtiQwertyKeyMappings*>(maps);
				
				if (overridingDeadKey != 0) 
					{				
					qmap->ClearDeadKey();
					}				
				else if (qmap->DeadKeyRootFlag())
					{
					if (iTextBuffer->Length() < iTextBuffer->MaxLength())
						{	
						SetFlag(EPtiDeadKeyRootFlag); // This will indicate PtiEngine that extra character 
						                             // was added to buffer.
                        chr = maps->StartMapping(aKey, iOwner->Case());
                        if ( chr )
                            {
                            iTextBuffer->Append( chr );
                            }
						}
					}
				else if ((qmap->VowelSequenceFlag()) &&
					     (lang->LanguageCode() == ELangVietnamese))
					{
					// This was Vietnamese vowel sequence, remove first vowel
					// component from buffer.
					TInt numRemove = (iTextBuffer->Length() > 1 ? 2 : 1);
					iTextBuffer->SetLength(iTextBuffer->Length() - numRemove);
					iTextBuffer->Append(qmap->VowelSequenceResult());						
					SetFlag(EPtiFlagVowelSeq);					
					}
				}				
			}
		}
		
	const TPtiTextCase textCase = iOwner->Case();		
// In Querty mode . default core always get called.
// When Fn key is pressed its not in multitap its  not in multitapmode
// multitap timer should be canceled.

	if (qwerty &&			
		((textCase == EPtiCaseLower) || (textCase == EPtiCaseUpper) ||(textCase == EPtiCaseFnLower) || (textCase == EPtiCaseFnUpper)||
		 ((inputMode == EPtiEngineZhuyinQwerty || inputMode ==EPtiEnginePinyinQwerty ||
		   inputMode == EPtiEngineStrokeQwerty ||inputMode == EPtiEngineZhuyinPhraseQwerty ||
		   inputMode == EPtiEngineStrokePhraseQwerty ||inputMode == EPtiEnginePinyinPhraseQwerty ) && 
		   (textCase == EPtiCaseChrUpper ||
		 textCase == EPtiCaseChrLower)))) 
		{
		iTimer->Cancel();
		}
	else 
		{
		iTimer->After(KMultiTapTimeoutMS); 
		}
	}



// ---------------------------------------------------------------------------
// CPtiDefaultCore::DeleteKeyPress
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::DeleteKeyPress()
	{
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));

	ClearFlag(EPtiGetAllFlag | EPtiFlagVowelSeq);		

	if (iTimer->IsActive())
		{
		iTimer->Cancel();
		}

	if (iTextBuffer->Length() > 0)
		{
		// Delete last character.
		iTextBuffer->SetLength(iTextBuffer->Length() - 1);	
		}

	ClearVowelSequence();
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::ClearBuffer
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::ClearBuffer()
	{
	__ASSERT_DEBUG(iTimer, User::Panic(KPtiEnginePanic, KErrCorrupt));
	
	iTimer->Cancel();	
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::ClearVowelSequence
// 
// ---------------------------------------------------------------------------
// 
void CPtiDefaultCore::ClearVowelSequence()
	{
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	if (iOwner->IsQwertyBasedMode(iOwner->InputMode()))
		{
		CPtiCoreLanguage* lang = static_cast<CPtiCoreLanguage*>(iOwner->CurrentLanguage());
		__ASSERT_DEBUG(lang, User::Panic(KPtiEnginePanic, KErrCorrupt));
		CPtiQwertyKeyMappings* qMap = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());
		if (qMap)
			{
			qMap->ClearVowelSequence();
			qMap->ClearLastChar();
			}
		}
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::SetCurrentInputSequence
// 
// ---------------------------------------------------------------------------
// 
TInt CPtiDefaultCore::SetCurrentInputSequence(TPtrC8 /*aInput*/)
	{
	__ASSERT_DEBUG(iTimer, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	if (iTimer->IsActive())
		{
		iTimer->Cancel();
		}	

	ClearVowelSequence();

	return KErrNone;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::TimerExpire
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::TimerExpire(TAny* aPtr)
	{
	return reinterpret_cast<CPtiDefaultCore*>(aPtr)->DoTimerExpire();
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::DoTimerExpire
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::DoTimerExpire()
	{
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));

	if (iOwner->Observer())
		{
		iOwner->Observer()->KeyTimerExpired();
		}
	return KErrNone;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::DropLanguages
// 
// ---------------------------------------------------------------------------
//
void CPtiDefaultCore::DropLanguages(CArrayPtrFlat<CPtiCoreLanguage>* aLanguageList)
	{
	for (TInt i = 0; i < aLanguageList->Count(); i++)
		{
		if (aLanguageList->At(i)->HasInputMode(EPtiEngineMultitapping))
			{
			aLanguageList->At(i)->ClearMultitapping();			
			if (aLanguageList->At(i)->GetCore(EPtiEngineMultitapping) == this)
			    {
			    aLanguageList->At(i)->SetCore(NULL, EPtiEngineMultitapping);			    
			    }
			}
        if (aLanguageList->At(i)->HasInputMode(EPtiEngineQwerty))			
            {
	        aLanguageList->At(i)->ClearQwerty(); 
		    if (aLanguageList->At(i)->GetCore(EPtiEngineQwerty) == this)
			    {
			    aLanguageList->At(i)->SetCore(NULL, EPtiEngineQwerty);			    
			    }	                   
            }
        if (aLanguageList->At(i)->HasInputMode(EPtiEngineHalfQwerty))			
            {
            aLanguageList->At(i)->ClearHalfQwerty();             
		    if (aLanguageList->At(i)->GetCore(EPtiEngineHalfQwerty) == this)
			    {
			    aLanguageList->At(i)->SetCore(NULL, EPtiEngineHalfQwerty);			    
			    }	                      
            }            
		}
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::GetCoreInfo
// 
// ---------------------------------------------------------------------------
//
MPtiCoreInfo* CPtiDefaultCore::GetCoreInfo()
	{
	return &iCoreInfo;   
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::Convert
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::Convert(TPtiCharConversion /*aType*/,
							  TAny* /*aInput*/,
							  TInt /*aInputLength*/,
							  TAny* /*aOutput*/)
	{
	return KErrNotSupported;	
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::SetExternalKeyMapL
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::SetExternalKeyMapL(TPtiEngineInputMode aMode,
		                                 TPtiKey aKey,
									     TDesC& aKeyMap,
										 TPtiTextCase aCase)
	{
	if (aMode == EPtiEngineNumeric)
		{
		__ASSERT_DEBUG(iNumericMappings, User::Panic(KPtiEnginePanic, KErrCorrupt));

		// Always available. Case doesn't apply for numeric mode,
		// so both cases are always updated.
		TInt ret = iNumericMappings->ReplaceKeyMapL(aKey, aKeyMap, EPtiCaseLower);
		if (ret != KErrNone)
			{
			return ret;
			} 
		return iNumericMappings->ReplaceKeyMapL(aKey, aKeyMap, EPtiCaseUpper);
		}
	else
		{		
		__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));
				
		CPtiCoreLanguage* lang = static_cast<CPtiCoreLanguage*>(iOwner->CurrentLanguage());
		if (!lang)
			{
			return KErrCorrupt;
			}
		
		MPtiKeyMappings* maps = NULL;
	    if (aMode == EPtiEngineMultitapping)
			{
			maps = lang->GetKeymappings();
			}
		else if (aMode == EPtiEngineQwerty)
			{
			maps = lang->GetQwertyKeymappings();		
			}		
        else if (aMode == EPtiEngineHalfQwerty)
            {
           	maps = lang->GetHalfQwertyKeymappings();	
            }
        						
		if (maps)
			{
			return maps->ReplaceKeyMapL(aKey, aKeyMap, aCase);
			}					
		}		
		

	return KErrNotSupported;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::GetSpelling
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::GetSpelling(TUint16 /*aInput*/, TDes& /*aOutput*/, 
                                  TPtiSpelling /*aType*/)
	{
	return KErrNotSupported;
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::CancelTimerActivity
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::CancelTimerActivity()
	{
	__ASSERT_DEBUG(iTimer, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	iTimer->Cancel();
	return KErrNone;	
	}


// ---------------------------------------------------------------------------
// CPtiDefaultCore::HandleCommandL
// 
// ---------------------------------------------------------------------------
//
TInt CPtiDefaultCore::HandleCommandL(TPtiEngineCommand aCommand, TAny* aParams)
	{
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));
		
	CPtiCoreLanguage* lang = static_cast<CPtiCoreLanguage*>(iOwner->CurrentLanguage());	
	TBool qwerty = EFalse;
	if (lang)
		{
		qwerty = lang->HasInputMode(EPtiEngineQwerty);
		}
	
	switch (aCommand)
		{
		case EPtiCommandPollQwertyDeadKeyRootFlag:
			{
			if (IsFlagSet(EPtiDeadKeyRootFlag))
				{
				ClearFlag(EPtiDeadKeyRootFlag);	
				return ETrue;
				}

			return EFalse;
			}
		case EPtiCommandGetAndClearDeadKeyRootChar:	
			{
			TUint16 ret = 0;
		
			if (lang && qwerty)
				{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());	
				if (maps->DeadKey())
					{
					maps->ClearDeadKey();	
					ret = maps->DeadKeyRootChar();						
					}								
				}
				
			return ret;
			}
		case EPtiCommandDeadKeyWaiting:	
			{
			TUint16 ret = 0;
		
			if (lang && qwerty)
				{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());	
				if (maps->DeadKey())	
					{
					ret = 1;						
					}								
				}
				
			return ret;
			}		
	
		case EPtiCommandQueryAndClearGetAllFlag:
			{
			TBool ret = IsFlagSet(EPtiGetAllFlag);
			ClearFlag(EPtiGetAllFlag);				
			return ret;	
			}
			
		case EPtiCommandGetAndClearVowelSequence:
			{
			if (IsFlagSet(EPtiFlagVowelSeq))
				{
				if (lang && qwerty)
					{
					CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());	
					ClearFlag(EPtiFlagVowelSeq);						
					return maps->VowelSequenceResult();						
					}								
				}
													
			return 0;
			}							
		case EPtiCommandVowelSeqFlag:
             return IsFlagSet(EPtiFlagVowelSeq);			
		case EPtiCommandClearVowelSeq:
			 {					
			 ClearFlag(EPtiFlagVowelSeq);
		     if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{
					maps->ClearVowelSequence();	
					}
				}					 
			 }
		     return 0; 		     
		case EPtiCommandGetAndClearLastVietnameseChar:
			 {					
			 if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{
					TInt ret = maps->GetLastChar();
					maps->ClearLastChar();	
					return ret;
					}
				}	 					
			 }
		     return 0;
		case EPtiCommandSetVietnameseLastChar:
	 		 {					
		     if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{
					maps->SetLastChar((TInt)aParams);	
					}
				}	 					
			 }					 
		     return 0;  
		case EPtiCommandQueryReplacePrevious:
			 {										 			 	 
			if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{
					TChar rep = maps->ReplacedCharacter();
					if (maps->VowelCase() == EPtiCaseUpper ||
					    maps->VowelCase() == EPtiCaseChrUpper)
						{
						rep.UpperCase();
						}
					else
						{
						rep.LowerCase();
						}						
					
					return rep;	
					}
				}	 					
			}				     
		     return 0;     	     		     		     		     
		case EPtiCommandSetLastKeyForVietnamese:
			 {	 		 			 	 
		     if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{					
					TPtiKey key = (TPtiKey)((TInt)aParams);
					maps->SetLastKey(key);	
					}
				}	 					
		 	 }
		 	 return 0;
		 	 
		case EPtiCommandResetVietnameseVowelSeqAndToneMark: 		  		     
		     ClearFlag(EPtiFlagVowelSeq);
			 if (lang && qwerty)
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{					
					maps->ResetVietnameseVowelSequenceAndToneMarks();	
					}
				}
			return 0;
			
		case EPtiCommandGetVietnameseLastKey:
		 	 if ((lang) && (lang->HasInputMode(EPtiEngineQwerty)))
			 	{
				CPtiQwertyKeyMappings* maps = static_cast<CPtiQwertyKeyMappings*>(lang->GetQwertyKeymappings());							
				if (maps)
					{					
					return (TInt)maps->LastKey();	
					}
				}				 			
		    return 0;
		case EPtiCommandDoubleWideEntry:
		    {
		    TInt ret = IsFlagSet(EPtiDoubleWideEntry);	
			ClearFlag(EPtiDoubleWideEntry);		    		
            return ret;
		    }		    			 											
		default:
			break;
		}		
	
	return KErrNotSupported;
	}
	
	
// ---------------------------------------------------------------------------
// CPtiDefaultCore::ListDataFactoryLanguagesL
// 
// ---------------------------------------------------------------------------
//	
void CPtiDefaultCore::ListDataFactoryLanguagesL(CPtiKeyMapDataFactory* aDataFactory,
                                                TInt aImplUid,
                                                CArrayPtrFlat<CPtiCoreLanguage>* aLanguageList)
    {
	__ASSERT_DEBUG(aDataFactory, User::Panic(KPtiEnginePanic, KErrCorrupt));
            
    RArray<TInt> langs;
    aDataFactory->ListLanguagesL(langs);    
    CleanupClosePushL(langs);            
                    
    for (TInt i = 0; i < langs.Count(); i++)
        {
        CPtiCoreLanguage* language = NULL;        
        
        for (TInt j = 0; j < iFactoryBindings.Count(); j++)    
            {
            // Remove possible earlier bindings for this language (ie. 
            // newly installed dlls will replace old languages).
            if (iFactoryBindings[j].iLanguage == langs[i])
                {
                iFactoryBindings.Remove(j);
                break;
                }          
            }
            
        if (!language)            
            {
            language = CPtiCoreLanguage::NewL();
	        CleanupStack::PushL( language );            
	       	aLanguageList->AppendL( language ); 
	        CleanupStack::Pop(); // language   	        
            }
            
		language->SetLanguageCode(langs[i]); 		
		language->SetCore(this, EPtiEngineNumeric);	
		language->SetKeyMapDataFactory(aDataFactory);
		language->GetKeymappings();  // Force core list to be updated.
		language->SetKeyMapDataFactory(NULL);		
            
        TPtiDataFactoryBinding newBind(langs[i], aImplUid);
        iFactoryBindings.AppendL(newBind);                       
        }
    
    CleanupStack::PopAndDestroy(); // langs    
    }
		 	
			
// ---------------------------------------------------------------------------
// CPtiDefaultCore::LoadFactoryForLanguageL
// 
// ---------------------------------------------------------------------------
//		
void CPtiDefaultCore::LoadFactoryForLanguageL(TInt aLanguage)
    {
	__ASSERT_DEBUG(iOwner, User::Panic(KPtiEnginePanic, KErrCorrupt));
        
    TInt currentUid = 0;    
    if (iKeyMapDataFactory)    
        {
        currentUid = iKeyMapDataFactory->ImplementationUid();
        }
        
    CPtiCoreLanguage* coreLang = static_cast<CPtiCoreLanguage*>(iOwner->GetLanguage(aLanguage));
    if (!coreLang)        
        {
        User::Leave(KErrCorrupt);
        }    
            
    coreLang->SetKeyMapDataFactory(NULL);            
    for (TInt i = 0; i < iFactoryBindings.Count(); i++)    
        {
        if (iFactoryBindings[i].iLanguage == aLanguage)
            {
            if (iFactoryBindings[i].iFactoryImplUid == currentUid)
                {
                coreLang->SetKeyMapDataFactory(iKeyMapDataFactory);               
                return;
                }
            
            // Factory is about to change. Clear references to old one.
            ShutDownKeymapFactory();
            
    	    iKeyMapDataFactory = CPtiKeyMapDataFactory::CreateImplementationL(TUid::Uid(iFactoryBindings[i].iFactoryImplUid));	  	    	  	   
            coreLang->SetKeyMapDataFactory(iKeyMapDataFactory);       	    
            
            return;            
            }
        }
        
    User::Leave(KErrNotFound);
    }
    

// ---------------------------------------------------------------------------
// CPtiDefaultCore::ShutDownKeymapFactory
// 
// ---------------------------------------------------------------------------
//	    
void CPtiDefaultCore::ShutDownKeymapFactory()
    {
    if (iLanguageList)
        {    
        CPtiCoreLanguage* lang = NULL;    	   
        for (TInt i = 0; i < iLanguageList->Count(); i++)
            {
    	    lang = static_cast<CPtiCoreLanguage*>(iLanguageList->At(i));
    	    if (lang && lang->DataFactory() == iKeyMapDataFactory)
    	        {
    		    lang->SetKeyMapDataFactory(NULL);		
    	        }
            }

        delete iKeyMapDataFactory;
        iKeyMapDataFactory = NULL;
        }   
    }
	
// End of file