os/graphics/windowing/windowserver/nga/SERVER/EVENT.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 1994-2010 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Top level window server code
    15 // 
    16 //
    17 
    18 #include "EVENT.H"
    19 
    20 #include "W32STD.H"
    21 #include <e32uid.h>
    22 #include <hal.h>
    23 #include <w32adll.h>
    24 #include "W32CLICK.H"
    25 #include "server.h"
    26 #include "windowgroup.h"
    27 #include "KEYCLICK.H"
    28 #include "wstop.h"
    29 #include "panics.h"
    30 #include "screen.h"
    31 #include "inifile.h"
    32 #include "password.h"
    33 #include "pointer.h"
    34 #include "debugbar.h"
    35 #include "advancedpointereventhelper.h"
    36 #include "graphics/wsgraphicdrawerinternal.h"
    37 #include "debughelper.h"
    38 
    39 GLREF_D CDebugLogBase *wsDebugLog;
    40 
    41 GLREF_C void StateDump();
    42 GLREF_C void HeapDump();
    43 
    44 _LIT(KDefaultKeyRouterPluginName, "keyrouter.dll");
    45 _LIT(KWSERVIniFileVarKeyRouterPlugin, "KEYROUTERPLUGIN");
    46 
    47 #define IMPOSSIBLE 0xFFFFFFFF
    48 
    49 const TWsWinCmdCaptureKey ImpossibleKeyPress=
    50 	{
    51 	IMPOSSIBLE,	// Impossible to hit key combination, used for disabling Hot Keys
    52 	IMPOSSIBLE,
    53 	IMPOSSIBLE};
    54 
    55 const TWsWinCmdCaptureKey DefaultHotKeys[TWindowServerEvent::ENumHotKeys]={
    56 	{	// Enable logging
    57 	EModifierFunc|EModifierCtrl|EModifierShift,
    58 	EModifierFunc|EModifierCtrl|EModifierShift,
    59 	5}, // E
    60 	{	// Disable logging
    61 	EModifierFunc|EModifierCtrl|EModifierShift,
    62 	EModifierFunc|EModifierCtrl|EModifierShift,
    63 	4}, // D
    64 	{	// Window server internal dump to log
    65 	EModifierFunc|EModifierCtrl|EModifierShift,
    66 	EModifierFunc|EModifierCtrl|EModifierShift,
    67 	23},// W
    68 	{	// The key of death
    69 	EModifierFunc|EModifierCtrl|EModifierShift,
    70 	EModifierFunc|EModifierCtrl|EModifierShift,
    71 	11},// K
    72 	{	// Shutdown window server
    73 #if defined(_DEBUG)
    74 	EModifierFunc|EModifierCtrl|EModifierShift,
    75 	EModifierFunc|EModifierCtrl|EModifierShift,
    76 	24},// X
    77 #else
    78 	IMPOSSIBLE,	// Impossible to hit key combination, effectively disables shutdown key in release builds
    79 	IMPOSSIBLE,
    80 	IMPOSSIBLE},
    81 #endif
    82 	{	// Heap dump
    83 	EModifierFunc|EModifierCtrl|EModifierShift,
    84 	EModifierFunc|EModifierCtrl|EModifierShift,
    85 	8}, // H
    86 	{	// Inc Contrast
    87 	0,
    88 	0,
    89 	EKeyIncContrast},
    90 	{	// Dec Contrast
    91 	0,
    92 	0,
    93 	EKeyDecContrast},
    94 	{	// Off
    95 	0,
    96 	0,
    97 	EKeyOff},
    98 	{	// Backlight on 
    99 	0,
   100 	0,
   101 	EKeyBacklightOn},
   102 	{	// Backlight off
   103 	0,
   104 	0,
   105 	EKeyBacklightOff},
   106 	{	// Backlight toggle
   107 	0,
   108 	0,
   109 	EKeyBacklightToggle},
   110 	{	// Screen Dimension Change
   111 	0,
   112 	0,
   113 	EKeyScreenDimension0},
   114 	{
   115 	0,
   116 	0,
   117 	EKeyScreenDimension1},
   118 	{
   119 	0,
   120 	0,
   121 	EKeyScreenDimension2},
   122 	{
   123 	0,
   124 	0,
   125 	EKeyScreenDimension3},
   126 #if defined(_DEBUG)
   127 	{	// Display mode cycle
   128 	EModifierFunc|EModifierCtrl|EModifierShift,
   129 	EModifierFunc|EModifierCtrl|EModifierShift,
   130 	21},// U
   131 	{	// Orientation cycle
   132 	EModifierFunc|EModifierCtrl|EModifierShift,
   133 	EModifierFunc|EModifierCtrl|EModifierShift,
   134 	15},// O
   135 #else
   136 	{	// Display mode cycle
   137 	IMPOSSIBLE,	// Impossible to hit key combination
   138 	IMPOSSIBLE,
   139 	IMPOSSIBLE},
   140 	{	// Orientation cycle
   141 	IMPOSSIBLE,	// Impossible to hit key combination
   142 	IMPOSSIBLE,
   143 	IMPOSSIBLE},
   144 #endif
   145 	{	// Inc Brightness
   146 	0,
   147 	0,
   148 	EKeyIncBrightness},
   149 	{	// Dec Brightness
   150 	0,
   151 	0,
   152 	EKeyDecBrightness},
   153 	{	// Cycle focus screen
   154 	EModifierFunc|EModifierCtrl|EModifierShift,
   155 	EModifierFunc|EModifierCtrl|EModifierShift,
   156 	9}, // I
   157 	};
   158 
   159 CKeyTranslator *TWindowServerEvent::iKeyTranslator=NULL;
   160 TEventRequestQueue TWindowServerEvent::iSwitchOnQueue;
   161 TEventRequestQueue TWindowServerEvent::iErrorMessageQueue;
   162 TEventRequestQueue TWindowServerEvent::iModifierChangedQueue;
   163 TEventRequestQueue TWindowServerEvent::iGroupChangedQueue;
   164 TEventRequestQueue TWindowServerEvent::iFocusChangedQueue;
   165 TEventRequestQueue TWindowServerEvent::iGroupListChangedQueue;
   166 TEventRequestQueue TWindowServerEvent::iScreenDeviceChangedQueue;
   167 TTime TWindowServerEvent::iPrevOomMessageTime;
   168 CCaptureKeys *TWindowServerEvent::iCaptureKeys;
   169 CKeyEventRouter* TWindowServerEvent::iKeyEventRouter;
   170 RLibrary TWindowServerEvent::iKeyEventRouterLibrary;
   171 CWsHotKey *TWindowServerEvent::iHotKeys;
   172 TInt TWindowServerEvent::iModifierState;
   173 CRawEventReceiver *TWindowServerEvent::iEventReceiver;
   174 RArray<TWindowServerEvent::TRawEventHandler> TWindowServerEvent::iEventHandlers;
   175 CArrayFixFlat<SNotificationHandler> *TWindowServerEvent::iNotificationHandlers;
   176 TInt TWindowServerEvent::iPotentialEventHandlers=0;
   177 TUint32 TWindowServerEvent::iBinaryFlags=0x00;
   178 RArray<TDrawerHandler>* TWindowServerEvent::iDrawerHandlers;
   179 RArray<TWsEventHandler> TWindowServerEvent::iWsEventHandlers;
   180 TInt TWindowServerEvent::iEventHandlerCount=0;
   181 TRepeatKey CKeyboardRepeat::iCurrentRepeat;
   182 TRepeatKey CKeyboardRepeat::iAlternateRepeat;
   183 TRepeatKey CKeyboardRepeat::iLongRepeat;
   184 TInt CKeyboardRepeat::iRepeatRollover=1;
   185 CKeyboardRepeat::TRepeatType CKeyboardRepeat::iRepeating=ERepeatNone;
   186 CKeyboardRepeat *CKeyboardRepeat::iThis=NULL;
   187 TTimeIntervalMicroSeconds32 CKeyboardRepeat::iInitialTime;
   188 TTimeIntervalMicroSeconds32 CKeyboardRepeat::iTime;
   189 TBool CKeyboardRepeat::iAlternateRepeatExists=EFalse;
   190 CWsCaptureLongKey* CKeyboardRepeat::iLongCapture=NULL;
   191 
   192 
   193 void TWindowServerEvent::DeleteHotKeys()
   194 	{
   195 	CWsHotKey *hotKey=iHotKeys;
   196 	while(hotKey)
   197 		{
   198 		CWsHotKey *next=hotKey->iNext;
   199 		delete hotKey;
   200 		hotKey=next;
   201 		}
   202 	iHotKeys=NULL;
   203 	}
   204 
   205 void TWindowServerEvent::DeleteStatics()
   206 	{
   207 	DeleteHotKeys();
   208 	delete iCaptureKeys;
   209 	delete iKeyEventRouter;
   210 	iKeyEventRouterLibrary.Close();
   211 	CKeyboardRepeat::Destroy();
   212 	delete iKeyTranslator;
   213 	delete iEventReceiver;
   214 	iEventHandlers.Close();
   215 	delete iNotificationHandlers;
   216 	iDrawerHandlers->Close();
   217 	delete iDrawerHandlers;
   218 	iWsEventHandlers.Close();
   219 	}
   220 
   221 void TWindowServerEvent::InitStaticsL()
   222 //
   223 // Create the CEvent active object.
   224 //
   225 	{
   226 #if defined(__WINS__)
   227 	WS_ASSERT_DEBUG(TWindowServerEvent::ENumHotKeys==EHotKeyLastKeyType+1, EWsPanicUnknownCaptureKey);
   228 #endif
   229 	iEventReceiver=new(ELeave) CRawEventReceiver(EEventPriority);
   230 	iEventReceiver->ConstructL();
   231 	iKeyTranslator=CKeyTranslator::New();
   232 	User::LeaveIfNull(iKeyTranslator);
   233     
   234 //  Change keyboard mapping according to information the HAL
   235 	TInt keyboardIndex;
   236 	if (HAL::Get(HALData::EKeyboardIndex,keyboardIndex)==KErrNone)
   237 		{
   238 		_LIT(KLitKeyDataDllName,"EKDATA.%02d");
   239 		TBuf<16> keyDataDllName;
   240 		keyDataDllName.Format(KLitKeyDataDllName,keyboardIndex);
   241 		iKeyTranslator->ChangeKeyData(keyDataDllName);
   242 		}
   243 
   244 	// CCaptureKeys is no longer used but a dummy object is required for
   245 	// calls to CKeyTranslator::TranslateKey() until capture functionality
   246 	// has been removed from ektran.dll.
   247 	iCaptureKeys=new(ELeave) CCaptureKeys;
   248 	iCaptureKeys->Construct();
   249 
   250 	// Load the key event routing plug-in. The DLL name may be overridden
   251 	// by setting the keyword KEYROUTERPLUGIN in wsini.ini.
   252 	TPtrC pluginName(KDefaultKeyRouterPluginName);
   253 	WsIniFile->FindVar(KWSERVIniFileVarKeyRouterPlugin, pluginName);
   254 	const TUidType uidType(KDynamicLibraryUid, KKeyRouterPluginUid);
   255 	TInt err = iKeyEventRouterLibrary.Load(pluginName, uidType);
   256 
   257 	if (wsDebugLog)
   258 		{
   259 		TLogMessageText buf;
   260 
   261 		if (err == KErrNone)
   262 			{
   263 			_LIT(KLogLoadOk, "Loaded plugin '%S' UID3 0x%x");
   264 			const TFileName& pluginPathname = iKeyEventRouterLibrary.FileName();
   265 			const TUid uid3 = iKeyEventRouterLibrary.Type()[2];
   266 			buf.Format(KLogLoadOk, &pluginPathname, uid3.iUid);
   267 			}
   268 		else
   269 			{
   270 			_LIT(KLogLoadError, "Failed to load plugin '%S' (error %d)");
   271 			buf.Format(KLogLoadError, &pluginName, err);
   272 			}
   273 
   274 		wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant, buf);
   275 		}
   276 
   277 	if (err != KErrNone)
   278 		{
   279 #ifdef _DEBUG
   280 		_LIT(KLoadError, "WServ: failed to load plugin '%S' (error %d)");
   281 		RDebug::Print(KLoadError, &pluginName, err);
   282 #endif
   283 		User::Leave(err);
   284 		}
   285 
   286 	// Create the key event router
   287 	typedef CKeyEventRouter* (*TCreateFunc)();
   288 	TCreateFunc newL = reinterpret_cast<TCreateFunc>(iKeyEventRouterLibrary.Lookup(1));
   289 	if (newL == NULL)
   290 		{
   291 		User::Leave(KErrNotFound);
   292 		}
   293 	iKeyEventRouter = (*newL)();
   294 
   295 	for (TInt index=0;index<TWindowServerEvent::ENumHotKeys;index++)
   296 		ConstructDefaultHotKeyL(index,DefaultHotKeys[index]);
   297 	CKeyboardRepeat::NewL();
   298 	CKeyboardRepeat::SetRepeatTime(EDefaultInitialRepeatTime, EDefaultRepeatTime);
   299 	iEventHandlers=RArray<TRawEventHandler>(2);
   300 	iNotificationHandlers=new(ELeave) CArrayFixFlat<SNotificationHandler>(2);
   301 	iDrawerHandlers = new(ELeave) RArray<TDrawerHandler>(4);
   302 	}
   303 
   304 void TWindowServerEvent::LinkHotKey(CWsHotKey *aWsHotKey)
   305 	{
   306 	aWsHotKey->SetLink(iHotKeys);
   307 	iHotKeys=aWsHotKey;
   308 	}
   309 
   310 void TWindowServerEvent::ConstructDefaultHotKeyL(TInt aHotKey, const TWsWinCmdCaptureKey &aSystemKey)
   311 	{
   312 	CWsHotKey* hotKey=new(ELeave) CWsHotKey(aHotKey, ETrue);
   313 	// hotKey is pushed onto the cleanup stack in method ConstructLD.
   314 	hotKey->ConstructLD(aSystemKey);
   315 	LinkHotKey(hotKey);
   316 	}
   317 
   318 CWsHotKey* TWindowServerEvent::ClearHotKeysL(TInt aHotKey)
   319 	{
   320 	if (aHotKey>ENumHotKeys)
   321 		{
   322 		User::Leave(KErrArgument);
   323 		}
   324 	CWsHotKey** pHotKey= &iHotKeys;
   325 	CWsHotKey* defaultHotKey=NULL;
   326 	while(*pHotKey)
   327 		{
   328 		TBool unlinked=EFalse;
   329 		if ((*pHotKey)->HotKeyType()==aHotKey)
   330 			{
   331 			CWsHotKey *free=*pHotKey;
   332 			if (free->IsDefault())
   333 				{
   334 				free->SetL(ImpossibleKeyPress);
   335 				defaultHotKey=free;
   336 				}
   337 			else
   338 				{
   339 				*pHotKey=(*pHotKey)->iNext;
   340 				delete free;
   341 				unlinked=ETrue;
   342 				}
   343 			}
   344 		if (!unlinked)
   345 			{
   346 			pHotKey=&(*pHotKey)->iNext;
   347 			}
   348 		}
   349 	return(defaultHotKey);
   350 	}
   351 
   352 void TWindowServerEvent::ResetDefaultHotKeyL(TInt aHotKey)
   353 	{
   354 	if ((aHotKey<0) || (aHotKey>=ENumHotKeys))
   355 		{
   356 		User::Leave(KErrArgument);
   357 		}
   358 	CWsHotKey* defaultHotKey=ClearHotKeysL(aHotKey);
   359 	WS_ASSERT_DEBUG(defaultHotKey, EWsPanicDefaultHotKeyNotFound);
   360 	defaultHotKey->SetL(DefaultHotKeys[aHotKey]);
   361 	}
   362 
   363 void TWindowServerEvent::SetHotKeyL(const TWsClCmdSetHotKey &aHotKey)
   364 	{
   365 	if (aHotKey.type>ENumHotKeys)
   366 		User::Leave(KErrArgument);
   367 //
   368 	CWsHotKey *hotKey=new(ELeave) CWsHotKey(aHotKey.type, EFalse);
   369 //
   370 	TWsWinCmdCaptureKey captureKey;
   371 	captureKey.modifiers=aHotKey.modifiers;
   372 	captureKey.modifierMask=aHotKey.modifierMask;
   373 	captureKey.key=aHotKey.keycode;
   374 	captureKey.priority = 0;
   375 	hotKey->ConstructLD(captureKey);
   376 //
   377 	LinkHotKey(hotKey);
   378 	}
   379 
   380 void TWindowServerEvent::AddEventHandler(MEventHandler *aEventHandler, TBool aAdvancedPointersEnabled)
   381 	{
   382 #if defined(_DEBUG)
   383 	TRAPD(err,iEventHandlers.AppendL(TRawEventHandler(aEventHandler, aAdvancedPointersEnabled)));
   384 	WS_ASSERT_DEBUG(err==KErrNone, EWsPanicEventHandlerInconsistency);
   385 #else
   386 	iEventHandlers.AppendL(TRawEventHandler(aEventHandler, aAdvancedPointersEnabled));	//Shouldn't leave
   387 #endif
   388 #ifdef LOG_WSERV_EVENTS
   389 	RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::AddEventHandler Added handler = %d AdvancedPointerEnabled = %d", iEventHandlers.Count(),aAdvancedPointersEnabled);
   390 #endif
   391 	}
   392 
   393 void TWindowServerEvent::RemoveEventHandler(const MEventHandler *aEventHandler)
   394 	{
   395 	TInt count=iEventHandlers.Count();
   396 	TInt ii;
   397 	for(ii=0;ii<count;++ii)
   398 		{
   399 		if (iEventHandlers[ii].iEventHandler==aEventHandler)
   400 			{
   401 #ifdef LOG_WSERV_EVENTS
   402 		    RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::RemoveEventHandler Removed handler = %d",ii);
   403 #endif
   404 			if (iEventHandlerCount>0)  
   405 				{
   406 				iBinaryFlags |= ERemovedEventHandlerWhileProcessingRawEvents;
   407 				iEventHandlers[ii].iEventHandler=NULL; // replace the Handler with null to keep size of the array
   408 				}
   409 			else 
   410 				{
   411 				iEventHandlers.Remove(ii);
   412 				}
   413 			return;
   414 			}
   415 		}
   416 	}
   417 
   418 void TWindowServerEvent::PotentialEventHandlerL(TInt aNum)
   419 	{
   420 	iPotentialEventHandlers+=aNum;
   421 	WS_ASSERT_DEBUG(iPotentialEventHandlers>=iEventHandlers.Count(), EWsPanicEventHandlerInconsistency);
   422 	TRAPD(err,iEventHandlers.Reserve(iPotentialEventHandlers));
   423 	if (err!=KErrNone)
   424 		{
   425 		if (aNum>0)
   426 			User::Leave(err);
   427 		}
   428 	else if (iPotentialEventHandlers==0)
   429 		iEventHandlers.Compress();
   430 	}
   431 
   432 void SendSwitchOnEvent(TEventRequestItem *aQptr, TInt aEvent, TInt )
   433 	{
   434 	aQptr->iWindow->QueueEvent(aEvent);
   435 	}
   436 
   437 /*void SendSwitchOffEvent(TEventRequestItem *aQptr, TInt , TInt )
   438 	{
   439 	aQptr->iWindow->QueueEvent(EEventSwitchOff);
   440 	}*/
   441 
   442 void SendErrorMessage(TEventRequestItem *aQptr, TInt aCategory, TInt aError)
   443 	{
   444 	TWsEvent event;
   445 	event.SetType(EEventErrorMessage);
   446 	event.SetHandle(aQptr->iWindow->ClientHandle());
   447 	event.ErrorMessage()->iErrorCategory=(TWsErrorMessage::TErrorCategory)aCategory;
   448 	event.ErrorMessage()->iError=aError;
   449 	event.SetTimeNow();
   450 	aQptr->iWindow->EventQueue()->QueueEvent(event,EEventPriorityHigh);
   451 	}
   452 
   453 void SendModifierChangedEvent(TEventRequestItem *aQptr, TInt aChanged, TInt )
   454 	{
   455 	TInt tmpChanged=aChanged&aQptr->iParam;
   456 	if (tmpChanged)
   457 		{
   458 		TWsEvent event;
   459 		event.SetType(EEventModifiersChanged);
   460 		event.SetHandle(aQptr->iWindow->ClientHandle());
   461 		event.ModifiersChanged()->iChangedModifiers=tmpChanged;
   462 		event.ModifiersChanged()->iModifiers=TWindowServerEvent::GetStoredModifierState();
   463 		event.SetTimeNow();
   464 		aQptr->iWindow->EventQueue()->QueueEvent(event,EEventPriorityHigh);
   465 		}
   466 	}
   467 
   468 void TWindowServerEvent::ProcessEventQueue(TEventRequestQueue &aQueue, TSendEventFunc aFunc, TInt aParam1, TInt aParam2)
   469 	{
   470 	TSglQueIter<TEventRequestItem> iter(aQueue.Queue());
   471 	TEventRequestItem *qPtr;
   472 	CWsWindowGroup *focusWin=CWsTop::FocusWindowGroup();
   473 	while((qPtr=iter++)!=NULL)
   474 		{
   475 		if (qPtr->iCircumstances==EEventControlAlways || 
   476 			(qPtr->iCircumstances==EEventControlOnlyWithKeyboardFocus && qPtr->iWindow->WinGroup()==focusWin) ||
   477 			(qPtr->iCircumstances==EEventControlOnlyWhenVisible && !qPtr->iWindow->TreeIsObscured()))
   478 			aFunc(qPtr, aParam1, aParam2);
   479 		}
   480 	}
   481 
   482 void TWindowServerEvent::NotifyOom()
   483 	{
   484 	TTime now;
   485 	now.UniversalTime();
   486 	TTimeIntervalSeconds interval;
   487 	TInt err=now.SecondsFrom(iPrevOomMessageTime,interval);
   488 	if (err!=KErrNone || interval.Int()<0 || interval.Int()>EOomEventSecondGap)
   489 		{
   490 		ProcessErrorMessages(TWsErrorMessage::EDrawingRegion,KErrNoMemory);
   491 		iPrevOomMessageTime=now;
   492 		}
   493 	}
   494 
   495 TBool TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::TErrorCategory aCategory, TInt aError)
   496 	{
   497 	if (aError!=KErrNone)
   498 		{
   499 		ProcessEventQueue(iErrorMessageQueue, SendErrorMessage, aCategory, aError);
   500 		return ETrue;
   501 		}
   502 	return EFalse;
   503 	}
   504 
   505 void TWindowServerEvent::ProcessModifierChanges()
   506 	{
   507 	TInt newState=iKeyTranslator->GetModifierState();
   508 	if (newState!=iModifierState)
   509 		{
   510 		TInt changed=iModifierState^newState;
   511 		iModifierState=newState;
   512 		ProcessEventQueue(iModifierChangedQueue, SendModifierChangedEvent, changed, 0);
   513 		}
   514 	}
   515 
   516 TEventQueueWalkRet FindScreenDeviceChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
   517 	{
   518 	if (aEvent->Type()==EEventScreenDeviceChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
   519 		*(TUint *)aHandlePtr=0;	// Indicates event found
   520 	return(EEventQueueWalkOk);
   521 	}
   522 	
   523 void TWindowServerEvent::SendScreenDeviceChangedEvents(CScreen* aScreen)
   524 	{
   525 	TSglQueIter<TEventRequestItem> iter(iScreenDeviceChangedQueue .Queue());
   526 	TEventRequestItem *qPtr;
   527 	while((qPtr=iter++)!=NULL)
   528 		SendScreenDeviceChangedEvent(qPtr->iWindow);
   529 	if(CClick::IsHandler())
   530 		{
   531 		TClickMakerData clickMakerData;
   532 		clickMakerData.screenDeviceMode=aScreen->ScreenSizeMode();
   533 		CClick::OtherEvent(EEventScreenDeviceChanged, &clickMakerData);
   534 		}
   535 	TWsEvent wsEvent;
   536 	wsEvent.SetType(EEventScreenDeviceChanged);
   537 	TWindowServerEvent::PublishNotification(wsEvent);
   538 	TWservCrEvent crEvent(TWservCrEvent::EScreenSizeModeChanged,aScreen->ScreenSizeMode());
   539 	TWindowServerEvent::NotifyDrawer(crEvent);
   540 	}
   541 
   542 void TWindowServerEvent::SendScreenDeviceChangedEvent(const CWsWindowBase *aWindow)
   543 	{
   544 	CEventQueue *queue=aWindow->EventQueue();
   545 	TUint32 handle=aWindow->ClientHandle();
   546 	queue->WalkEventQueue(&FindScreenDeviceChangedEvent,&handle);
   547 	if (handle!=NULL)	// Indicates event not found
   548 		queue->QueueEvent(handle, EEventScreenDeviceChanged);
   549 	}
   550 
   551 TEventQueueWalkRet FindGroupChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
   552 	{
   553 	if (aEvent->Type()==EEventWindowGroupsChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
   554 		{
   555 		*(TUint *)aHandlePtr=0;	// Indicates event found
   556 		}
   557 	return(EEventQueueWalkOk);
   558 	}
   559 	
   560 void TWindowServerEvent::SendGroupChangedEvents()
   561 	{
   562 	TSglQueIter<TEventRequestItem> iter(iGroupChangedQueue.Queue());
   563 	TEventRequestItem *qPtr;
   564 	while((qPtr=iter++)!=NULL)
   565 		{
   566 		const CWsWindowBase *win=qPtr->iWindow;
   567 		CEventQueue *queue=win->EventQueue();
   568 		TUint32 handle=win->ClientHandle();
   569 		queue->WalkEventQueue(&FindGroupChangedEvent,&handle);
   570 		if (handle!=NULL)	// Indicates event not found
   571 			{
   572 			queue->QueueEvent(handle, EEventWindowGroupsChanged);
   573 			}
   574 		}
   575 	}	
   576 
   577 TEventQueueWalkRet FindFocusChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
   578 	{
   579 	if (aEvent->Type()==EEventFocusGroupChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
   580 		{
   581 		*(TUint *)aHandlePtr=0;	// Indicates event found
   582 		}
   583 	return(EEventQueueWalkOk);
   584 	}
   585 	
   586 void TWindowServerEvent::SendFocusChangedEvents()
   587 	{
   588 	TInt identifier=0;	// Zero Identifier indicates, currently there is no focused window group
   589 	CScreen* currentFocusScreen=CWsTop::CurrentFocusScreen();	
   590 	TInt screenNumber=currentFocusScreen->ScreenNumber();
   591 	CWsWindowGroup* currentFocusWG=currentFocusScreen->FocusWindowGroup();
   592 	if(currentFocusWG)
   593 		{
   594 		identifier=currentFocusWG->Identifier();
   595 		}
   596 	TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EWindowGroupChanged,
   597 		screenNumber, reinterpret_cast<TAny*>(identifier)));
   598 		 
   599 	TSglQueIter<TEventRequestItem> iter(iFocusChangedQueue.Queue());
   600 	TEventRequestItem *qPtr;
   601 	while((qPtr=iter++)!=NULL)
   602 		{
   603 		const CWsWindowBase *win=qPtr->iWindow;
   604 		CEventQueue *queue=win->EventQueue();
   605 		TUint32 handle=win->ClientHandle();
   606 		queue->WalkEventQueue(&FindFocusChangedEvent,&handle);
   607 		if (handle!=NULL)	// Indicates event not found
   608 			{
   609 			queue->QueueEvent(handle, EEventFocusGroupChanged);
   610 			}
   611 		}
   612 	}
   613 
   614 TEventQueueWalkRet FindGroupListChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
   615 	{
   616 	if (aEvent->Type()==EEventWindowGroupListChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
   617 		{
   618 		*(TUint *)aHandlePtr=0;	// Indicates event found
   619 		}
   620 	return(EEventQueueWalkOk);
   621 	}
   622 	
   623 void TWindowServerEvent::SendGroupListChangedEvents()
   624 	{
   625 	TSglQueIter<TEventRequestItem> iter(iGroupListChangedQueue.Queue());
   626 	TEventRequestItem *qPtr;
   627 	while((qPtr=iter++)!=NULL)
   628 		{
   629 		const CWsWindowBase *win=qPtr->iWindow;
   630 		CEventQueue *queue=win->EventQueue();
   631 		TUint32 handle=win->ClientHandle();
   632 		queue->WalkEventQueue(&FindGroupListChangedEvent,&handle);
   633 		if (handle!=NULL)	// Indicates event not found
   634 			{
   635 			queue->QueueEvent(handle, EEventWindowGroupListChanged);
   636 			}
   637 		}
   638 	}	
   639 
   640 TEventQueueWalkRet OverrideVisibilityChangedEvent(TAny *aNewEvent, TWsEvent *aOldEvent)
   641 	{
   642 	// This replaces the first visibility event it finds for the given window with the
   643 	// one given.  This is fine, so long as the meaning of all visibility events remains
   644 	// independent of the ones before.
   645 	TWsEvent* newEvent = static_cast<TWsEvent*>(aNewEvent);
   646 	if (aOldEvent->Type()==EEventWindowVisibilityChanged && aOldEvent->Handle()==newEvent->Handle())
   647 		{
   648 		aOldEvent->SetTimeNow();
   649 		aOldEvent->VisibilityChanged()->iFlags = newEvent->VisibilityChanged()->iFlags;
   650 		newEvent->SetHandle(NULL);
   651 		}
   652 	return EEventQueueWalkOk;
   653 	}
   654 
   655 void TWindowServerEvent::SendVisibilityChangedEvents(CWsWindowBase* aWin, TUint aFlags)
   656 	{
   657 	CEventQueue *queue=aWin->EventQueue();
   658 	TWsEvent event;
   659 	event.SetType(EEventWindowVisibilityChanged);
   660 	event.SetHandle(aWin->ClientHandle());
   661 	event.SetTimeNow();
   662 	TWsVisibilityChangedEvent* visevent = event.VisibilityChanged();
   663 	visevent->iFlags = aFlags;
   664 	queue->WalkEventQueue(&OverrideVisibilityChangedEvent,&event);
   665 	if (event.Handle()!=NULL)
   666 		{
   667 		queue->QueueEvent(event);
   668 		}
   669 	}
   670 
   671 TEventQueueWalkRet OverrideDisplayChangedEvent(TAny *aNewEvent, TWsEvent *aOldEvent)
   672 	{
   673 	TWsEvent* newEvent = static_cast<TWsEvent*>(aNewEvent);
   674 	if (aOldEvent->Type() == EEventDisplayChanged && 
   675 			aOldEvent->DisplayChanged()->iDisplayNumber == newEvent->DisplayChanged()->iDisplayNumber)
   676 		{
   677 		aOldEvent->SetTimeNow();
   678 		aOldEvent->DisplayChanged()->iConfigurationChangeId = newEvent->DisplayChanged()->iConfigurationChangeId;
   679 		aOldEvent->DisplayChanged()->iResolutionListChangeId = newEvent->DisplayChanged()->iResolutionListChangeId;
   680 		newEvent->DisplayChanged()->iDisplayNumber = KErrNotFound; //So the new event won't be placed on event queue again
   681 		}
   682 	return EEventQueueWalkOk;
   683 	}
   684 
   685 TBool TWindowServerEvent::SendDisplayChangedEvents(CWsClient* aWsClient, TInt aDisplayNumber, TInt aConfigurationChangeId, TInt aResolutionListChangeId)
   686 	{
   687 	CEventQueue *queue = aWsClient->EventQueue();
   688 	TWsEvent event;
   689 	event.SetType(EEventDisplayChanged);
   690 	event.SetTimeNow();
   691 	
   692     // fill in the handle otherwise CONE will discard the notification
   693     CWsObjectIx* clientObjList = aWsClient->ObjectIndex();
   694     const TWsObject* ptr=clientObjList->FirstObject();
   695     const TWsObject* end=ptr+clientObjList->Length();
   696     while(++ptr<end)    // first one should always have a NULL object
   697         {
   698         const CWsObject* obj=ptr->iObject;
   699         if (obj && obj->Type()==WS_HANDLE_GROUP_WINDOW)
   700             {
   701             event.SetHandle(ptr->iHandle);
   702             break;
   703             }
   704         }	
   705 	
   706 	TWsDisplayChangedEvent* dispEvent = event.DisplayChanged();
   707 	dispEvent->iDisplayNumber = aDisplayNumber;
   708 	dispEvent->iConfigurationChangeId = aConfigurationChangeId;
   709 	dispEvent->iResolutionListChangeId = aResolutionListChangeId;
   710 	queue->WalkEventQueue(&OverrideDisplayChangedEvent, &event);
   711 	//place the new event on the queue only when its display number is valid (!=KErrNotFound)
   712 	if(event.DisplayChanged()->iDisplayNumber >= 0)
   713 		{
   714 		return queue->QueueEvent(event);
   715 		}
   716 	return ETrue;
   717 	}
   718 
   719 void TWindowServerEvent::QueueKeyEvent(CWsWindowGroup *aWin, TWsEvent &aEvent, TWservEventPriorities aPriority)
   720 	{
   721 #ifdef LOG_WSERV_EVENTS
   722     RDebug::Print(_L("_WSEVENT_KEY: TWindowServerEvent::QueueKeyEvent, Queuing event name %S for application read, window handle: %d"), &WsEventName(aEvent), CWsTop::FocusWindowGroup()->ClientHandle());
   723 #endif
   724 	aEvent.SetTimeNow();
   725 	aWin->EventQueue()->QueueEvent(aEvent, aPriority);
   726 	}
   727 
   728 /**
   729 Process a key press event.
   730 
   731 This function is called for every input key event and uses the Key Event
   732 Routing plug-in to check for short and long key capture and determine the
   733 destination window group for the queued event(s).
   734 Window server hotkeys are also processed.
   735 Note that the key repeat timer is started here but the key repeat events
   736 generated by the timer go directly to QueueKeyPress().
   737 
   738 @param	aKeyEvent		Input key event
   739 @param	aCheckRepeat	Check for key repeat and long key capture
   740 @param	aRepeats		Repeat count
   741 */
   742 void TWindowServerEvent::ProcessKeyPress(const TKeyEvent& aKeyEvent, TBool aCheckRepeat, TInt aRepeats)
   743  	{
   744 	CWsWindowGroup* focusWin = CWsTop::FocusWindowGroup();
   745 	TUid focusAppUid = focusWin ? TUid::Uid(focusWin->Client()->SecureId().iId) : KNullUid;
   746 
   747 	// Route the key event and check for short key capture.
   748 	// Note that the Key Routing plugin may translate or block key events.
   749 	TKeyEventRouterInput input(ECaptureTypeKey, aKeyEvent, focusWin, focusAppUid);
   750 	TKeyEventRouterOutput output;
   751 
   752 #ifdef _DEBUG
   753 	// RouteKey() must not fail. Check for leaves in case the plug-in
   754 	// is badly behaved.
   755 	TRAPD(err, iKeyEventRouter->RouteKey(input, output));
   756 	WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
   757 #else
   758 	iKeyEventRouter->RouteKey(input, output);
   759 #endif
   760 
   761 	WS_ASSERT_DEBUG(output.iResult == ERouted || output.iResult == ECaptured || output.iResult == EConsumed, EWsPanicKeyEventRouterBadResult);
   762 
   763 	if (output.iResult == EConsumed)
   764 		{
   765 		focusWin = NULL;
   766 		}
   767 	else
   768 		{
   769 		focusWin = static_cast<CWsWindowGroup*>(output.iWindowGroup);
   770 		}
   771 	WS_ASSERT_DEBUG((focusWin == NULL || focusWin->Type() == WS_HANDLE_GROUP_WINDOW) && (output.iResult != ERouted || focusWin == CWsTop::FocusWindowGroup()), EWsPanicKeyEventRouterBadWindowGroup);
   772 
   773 	// Ensure that short event is not marked with EModifierLongKey
   774 	output.iKeyEvent.iModifiers &= ~EModifierLongKey;
   775 
   776 	// Generate key click unless the event is consumed. This is consistent
   777 	// with the behaviour when CKeyTranslator::TranslateKey() yields no
   778 	// translation for a particular scan code. (Click events for key up/down
   779 	// will still be generated by QueueKeyUpDown()). Note however that a long
   780 	// key press may still be captured even if the short event is consumed.
   781 	if (CClick::IsHandler() && output.iResult != EConsumed)
   782 		{
   783 		output.iKeyEvent.iRepeats = aRepeats;
   784 		CClick::KeyEvent(EEventKey, output.iKeyEvent);
   785 		}
   786 
   787 	if (output.iResult == ECaptured)
   788 		{
   789 		if (output.iWindowGroup == NULL)	// Captured by Wserv itself
   790 			{
   791 			_LIT(KWSERVDebugLogCapturedKey,"WSERV Captured Key");
   792 			CScreen* focusScreen=CWsTop::CurrentFocusScreen();
   793 			TInt screenNo=focusScreen->ScreenNumber();
   794 			
   795 			if (wsDebugLog)
   796 				wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogCapturedKey);
   797 			CWsHotKey *hotKey=iHotKeys;
   798 			while(hotKey)
   799 				{
   800 				if (hotKey->KeyHandle() == reinterpret_cast<TInt>(output.iCaptureHandle))
   801 					{
   802 					switch(hotKey->HotKeyType())
   803 						{
   804 						case EHotKeyEnableLogging:
   805 							CWsTop::EnableLogging();
   806 							break;
   807 						case EHotKeyDisableLogging:
   808 							CWsTop::DisableLogging();
   809 							break;
   810 						case EHotKeyStateDump:
   811 							StateDump();
   812 							break;
   813 						case EHotKeyHeapDump:
   814 							HeapDump();
   815 							break;
   816 						case EHotKeyOfDeath:
   817 							if (!CWsPassword::PasswordModeActive())
   818 								{
   819 								const TBool currentJustInTimeValue=User::JustInTime();
   820 								if (currentJustInTimeValue)
   821 									{
   822 									User::SetJustInTime(EFalse);
   823 									}
   824 								CWsTop::KillForegroundSession();
   825 								if (currentJustInTimeValue)
   826 									{
   827 									User::SetJustInTime(ETrue);
   828 									}
   829 								}
   830 							break;
   831 						case EHotKeyShutDown:
   832 							CWsTop::Exit();
   833 							break;
   834 						case EHotKeyIncContrast:
   835 							focusScreen->IncContrast();
   836 							break;
   837 						case EHotKeyDecContrast:
   838 							focusScreen->DecContrast();
   839 							break;
   840 						case EHotKeyOff:
   841 							CWsTop::HandleSwitchOff(EEventKeySwitchOff,ETrue);
   842 							break;
   843 						case EHotKeyBacklightToggle:
   844 							{
   845 							TInt state;
   846 							if (!ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(screenNo,HALData::EBacklightState,state)))
   847 								ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,!state));
   848 							}
   849 							break;
   850 						case EHotKeyBacklightOn:
   851 							ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,ETrue));
   852 							break;
   853 						case EHotKeyBacklightOff:
   854 							ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,EFalse));
   855 							break;
   856 						case EHotKeyScreenDimension0:
   857 						case EHotKeyScreenDimension1:
   858 						case EHotKeyScreenDimension2:
   859 						case EHotKeyScreenDimension3:
   860 							focusScreen->doSetScreenMode(hotKey->HotKeyType()-EHotKeyScreenDimension0);
   861 							break;
   862 						case EHotKeyCycleDisplaySize:
   863 							focusScreen->CycleDisplaySize();
   864 							break;
   865 						case EHotKeyCycleOrientation:
   866 							focusScreen->CycleOrientation();
   867 							break;
   868 						case EHotKeyIncBrightness:
   869 							focusScreen->IncBrightness();
   870 							break;
   871 						case EHotKeyDecBrightness:
   872 							focusScreen->DecBrightness();
   873 							break;
   874 						case EHotKeyCycleFocusScreen:
   875 							CWsTop::SetCurrentFocusScreen((CWsTop::CurrentFocusScreen()->ScreenNumber()+1)%CWsTop::NumberOfScreens());
   876 							break;
   877 						}
   878 					return;
   879 					}
   880 				hotKey=hotKey->iNext;
   881 				}
   882 			WS_PANIC_ALWAYS(EWsPanicUnknownCaptureKey);
   883 			return;
   884 			}
   885 
   886 		_LIT(KWSERVDebugLogKeyCapturedByApp,"Key captured by app %d");
   887 		if (wsDebugLog)
   888 			wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyCapturedByApp,focusWin->Identifier());
   889 		if (CWsPassword::PasswordModeActive() && focusWin!=CWsPassword::PasswordWindow()->WinGroup())
   890 			return;
   891 		}
   892 
   893 	CWsCaptureLongKey* longCapture = NULL;
   894 	TKeyEventRouterOutput longOutput;
   895 	if (aCheckRepeat)
   896 		{
   897 		// Check for long key capture.
   898 		// Note that a long key event can only result from capture, there is
   899 		// no default detection or routing of long events.
   900 		input.iType = ECaptureTypeLongKey;
   901 #ifdef _DEBUG
   902 		TRAPD(err, iKeyEventRouter->RouteKey(input, longOutput));
   903 		WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
   904 #else
   905 		iKeyEventRouter->RouteKey(input, longOutput);
   906 #endif
   907 
   908 		if (longOutput.iResult == ECaptured)
   909 			{
   910 			longCapture = static_cast<CWsCaptureLongKey*>(longOutput.iCaptureHandle);
   911 
   912 			// Mark long key events with EModifierLongKey so that applications
   913 			// can easily distinguish short and long events.
   914 			longOutput.iKeyEvent.iModifiers |= EModifierLongKey;
   915 
   916 			// Start timer to detect long key press
   917 			CKeyboardRepeat::StartRepeat(aKeyEvent.iScanCode, output, &longOutput);
   918 			}
   919 		else if (output.iResult != EConsumed && output.iKeyEvent.iModifiers & EModifierAutorepeatable)
   920 			{
   921 			// Start timer for key repeat
   922 			CKeyboardRepeat::StartRepeat(aKeyEvent.iScanCode, output, NULL);
   923 			}
   924 		}
   925 
   926 	// Queue the short event
   927 	if (!longCapture || longCapture->iFlags & ELongCaptureShortEventImmediately)
   928 		{
   929 		QueueKeyPress(output, EFalse, aRepeats);
   930 		}
   931 	}
   932 
   933 /**
   934 Queue a key press event.
   935 
   936 This function is called for each key event produced by ProcessKeyPress(),
   937 for every key repeat and long key event generated by the timer and also for
   938 delayed short key events from KeyUp().
   939 
   940 @param	aOutput			Output key event from routing plug-in
   941 @param	aIsRepeat		Event is due to key repeat
   942 @param	aRepeats		Repeat count
   943 */
   944 void TWindowServerEvent::QueueKeyPress(const TKeyEventRouterOutput& aOutput, TBool aIsRepeat, TInt aRepeats)
   945  	{
   946 	if (aOutput.iResult == EConsumed)
   947 		{
   948 		// Don't deliver this key
   949 		return;
   950 		}
   951 
   952 	TWsEvent event;
   953 	TKeyEvent& keyEvent = *event.Key();
   954 	keyEvent = aOutput.iKeyEvent;
   955 	keyEvent.iRepeats = aRepeats;
   956 
   957 	CWsWindowGroup* focusWin = static_cast<CWsWindowGroup*>(aOutput.iWindowGroup);
   958 	WS_ASSERT_DEBUG(focusWin == NULL || focusWin->Type() == WS_HANDLE_GROUP_WINDOW, EWsPanicKeyEventRouterBadWindowGroup);
   959 
   960 	if (aIsRepeat && aOutput.iResult != ECaptured && focusWin != CWsTop::FocusWindowGroup())
   961 		CKeyboardRepeat::CancelRepeat(NULL);		// Repeat is going to different window so cancel it and don't deliver this key
   962 	else if (focusWin != NULL && focusWin->CheckForPriorityKey(keyEvent) == EFalse)
   963 		{
   964 		event.SetType(EEventKey);
   965 		event.SetHandle(focusWin->ClientHandle());
   966 		if (aRepeats!=0)
   967 			{
   968 			CEventQueue* queue=focusWin->EventQueue();
   969 			queue->Wait();
   970 			const TWsEvent* prev=queue->PeekLastEvent();
   971 			if (prev != NULL && prev->Type() == EEventKey && prev->Key()->iRepeats > 0 && prev->Key()->iCode == keyEvent.iCode)
   972 				{
   973 				prev->Key()->iRepeats += aRepeats;
   974 				queue->Signal();
   975 				if (CClick::IsHandler())
   976 					CClick::KeyEvent(EEventKeyRepeat, *prev->Key());
   977 				return;
   978 				}
   979 			queue->Signal();
   980 			if (CClick::IsHandler())
   981 				CClick::KeyEvent(EEventKeyRepeat,keyEvent);
   982 			}
   983 		QueueKeyEvent(focusWin, event, EEventPriorityLow);
   984 		}
   985 	}
   986 
   987 /**
   988 Queue a key up/down event.
   989 
   990 @param	aRawEvent		Raw event
   991 */
   992 void TWindowServerEvent::QueueKeyUpDown(const TRawEvent &aRawEvent)
   993  	{
   994 #ifdef LOG_WSERV_EVENTS
   995     RDebug::Print(_L("_WSEVENT_KEY: TWindowServerEvent::QueueKeyUpDown, Event Name: %S, Scan code: %d"), &RawEventName(aRawEvent), aRawEvent.ScanCode());
   996 #endif
   997 	TEventCode type = aRawEvent.Type() == TRawEvent::EKeyUp ? EEventKeyUp : EEventKeyDown;
   998 
   999 	// Check for key up/down capture
  1000 	TKeyEvent keyEvent;
  1001 	keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
  1002 #if defined(__WINS__)
  1003 	keyEvent.iCode = __WINS_CHARCODE(aRawEvent.ScanCode());
  1004 #else
  1005 	keyEvent.iCode = 0;
  1006 #endif
  1007 	keyEvent.iModifiers = iModifierState;
  1008 	keyEvent.iRepeats = 0;
  1009 
  1010 	CWsWindowGroup* focusWin = CWsTop::FocusWindowGroup();
  1011 	TUid focusAppUid = focusWin ? TUid::Uid(focusWin->Client()->SecureId().iId) : KNullUid;
  1012 
  1013 	TKeyEventRouterInput input(ECaptureTypeKeyUpDown, keyEvent, focusWin, focusAppUid);
  1014 	TKeyEventRouterOutput output;
  1015 #ifdef _DEBUG
  1016 	TRAPD(err, iKeyEventRouter->RouteKey(input, output));
  1017 	WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
  1018 #else
  1019 	iKeyEventRouter->RouteKey(input, output);
  1020 #endif
  1021 
  1022 	if (output.iResult == EConsumed)
  1023 		{
  1024 		// Don't deliver this key. A key click is still generated for the
  1025 		// input event.
  1026 		if (CClick::IsHandler())
  1027 			{
  1028 			CClick::KeyEvent(type, keyEvent);
  1029 			}
  1030 		return;
  1031 		}
  1032 	WS_ASSERT_DEBUG(output.iResult == ERouted || output.iResult == ECaptured, EWsPanicKeyEventRouterBadResult);
  1033 
  1034 	focusWin = static_cast<CWsWindowGroup*>(output.iWindowGroup);
  1035 	WS_ASSERT_DEBUG((focusWin == NULL || focusWin->Type() == WS_HANDLE_GROUP_WINDOW) && (output.iResult != ERouted || focusWin == CWsTop::FocusWindowGroup()), EWsPanicKeyEventRouterBadWindowGroup);
  1036 #if defined(__WINS__)
  1037 	if (focusWin && !focusWin->WsOwner()->RemoveKeyCode())
  1038 		{
  1039 		// Restore WINS character code
  1040 		output.iKeyEvent.iScanCode |= output.iKeyEvent.iCode;
  1041 		}
  1042 	output.iKeyEvent.iCode = 0;
  1043 #endif
  1044 
  1045 	output.iKeyEvent.iRepeats = 0;
  1046 	if (CClick::IsHandler())
  1047 		{
  1048 		CClick::KeyEvent(type, output.iKeyEvent);
  1049 		}
  1050 
  1051 	TWsEvent event;
  1052 	*event.Key() = output.iKeyEvent;
  1053 	if (focusWin!=NULL)
  1054 		{
  1055 		event.SetType(type);
  1056 		event.SetHandle(focusWin->ClientHandle());
  1057 		QueueKeyEvent(focusWin, event, EEventPriorityHigh);
  1058 		}
  1059 	}
  1060 
  1061 LOCAL_D void GetPointerEvent(TPointerEvent::TType& aType, const TRawEvent &aRawEvent, TBool& aHandled)
  1062 	{
  1063 	aHandled=ETrue;
  1064 	switch(aRawEvent.Type())
  1065 		{
  1066 	case TRawEvent::EButton1Down:
  1067 		aType=TPointerEvent::EButton1Down;
  1068 		break;
  1069 	case TRawEvent::EButton1Up:
  1070 		aType=TPointerEvent::EButton1Up;
  1071 		break;
  1072 	case TRawEvent::EButton2Down:
  1073 		aType=TPointerEvent::EButton2Down;
  1074 		break;
  1075 	case TRawEvent::EButton2Up:
  1076 		aType=TPointerEvent::EButton2Up;
  1077 		break;
  1078 	case TRawEvent::EButton3Down:
  1079 		aType=TPointerEvent::EButton3Down;
  1080 		break;
  1081 	case TRawEvent::EButton3Up:
  1082 		aType=TPointerEvent::EButton3Up;
  1083 		break;
  1084 	case TRawEvent::EPointerMove:
  1085 		aType=TPointerEvent::EMove;
  1086 		break;
  1087 	case TRawEvent::EPointerSwitchOn:
  1088 		aType=TPointerEvent::ESwitchOn;
  1089 		break;
  1090 	case TRawEvent::EPointer3DOutOfRange:
  1091 		aType=TPointerEvent::EOutOfRange;
  1092 		break;
  1093 	default:
  1094 		aHandled=EFalse;
  1095 		}
  1096 	}
  1097 
  1098 TBool TWindowServerEvent::MousePress(const TRawEvent &aRawEvent, const CWsWindowGroup *aGroupWin)
  1099 	//
  1100 	//Return EFalse if known not to be a Mouse Event
  1101 	//
  1102 	{
  1103 	TBool handled=ETrue;
  1104 	TPointerEvent::TType type;
  1105 	GetPointerEvent(type, aRawEvent, handled);
  1106 	if (handled)
  1107 		{
  1108 		TPoint3D point3D(0,0,0);
  1109 		if (type != TPointerEvent::EOutOfRange)
  1110 			{
  1111 			point3D = aRawEvent.Pos3D();
  1112 			}
  1113 		TWsEvent event;
  1114 		TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, type,iKeyTranslator->GetModifierState(),point3D,aRawEvent.PointerNumber());
  1115 		TWsPointer::ProcessWsEvent(event, aGroupWin, ETrue);
  1116 		}
  1117 	return handled;
  1118 	}
  1119 
  1120 LOCAL_D void SendEventToKeyClick(const TRawEvent& aRawEvent)
  1121 	{
  1122 	switch(aRawEvent.Type())
  1123 		{
  1124 		case TRawEvent::EKeyDown:
  1125 		case TRawEvent::EKeyUp:
  1126 			{
  1127 			TKeyEvent keyEvent;
  1128 			keyEvent.iCode=0;
  1129 			keyEvent.iScanCode=aRawEvent.ScanCode();
  1130 			keyEvent.iModifiers=0;
  1131 			keyEvent.iRepeats=0;
  1132 			CClick::KeyEvent(EEventKey,keyEvent);
  1133 			}
  1134 			break;
  1135 		case TRawEvent::EButton1Down:
  1136 		case TRawEvent::EButton1Up:
  1137 		case TRawEvent::EButton2Down:
  1138 		case TRawEvent::EButton2Up:
  1139 		case TRawEvent::EButton3Down:
  1140 		case TRawEvent::EButton3Up:
  1141 		case TRawEvent::EPointerMove:
  1142 		case TRawEvent::EPointerSwitchOn:
  1143 			{
  1144 			TBool handled=ETrue;
  1145 			TPointerEvent::TType type;
  1146 			GetPointerEvent(type, aRawEvent, handled);
  1147 			if (handled)
  1148 				{
  1149 				TWsEvent event;
  1150 				TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, type, 0, aRawEvent.Pos3D(), TPoint(), aRawEvent.PointerNumber());
  1151 				TAdvancedPointerEvent& pointerEvent = *event.Pointer();
  1152 				CClick::PointerEvent(pointerEvent.iPosition,pointerEvent);
  1153 				}
  1154 			}
  1155 			break;
  1156 		default:
  1157 			break;
  1158 		}
  1159 	}
  1160 
  1161 /*
  1162 Process a raw event
  1163 
  1164 @param	aRawEvent	Raw event
  1165 */
  1166 void TWindowServerEvent::ProcessRawEvent(const TRawEvent& aRawEvent)
  1167 //
  1168 // Event has completed.
  1169 //
  1170 	{
  1171 	TRawEvent::TType eventType = aRawEvent.Type();
  1172 	TBool isPointerEvent = TWsPointer::IsPointerEventType(eventType);
  1173 	if (isPointerEvent)
  1174 		{
  1175 #ifdef LOG_WSERV_EVENTS
  1176         RDebug::Print(_L("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent EventName = %S PointerNumber = %d PrimaryPointerNumber = %d Coordinates = ( %d, %d )"), 
  1177                 &RawEventName(aRawEvent),aRawEvent.PointerNumber(),TWsPointer::PrimaryPointer(),aRawEvent.Pos().iX,aRawEvent.Pos().iY); 
  1178 #endif
  1179         TWsPointer::UpdatePrimaryPointer(aRawEvent);
  1180 		}
  1181 	TInt count=iEventHandlers.Count();
  1182 	TInt ii;
  1183 	TBool eventHandled = EFalse;
  1184 	iEventHandlerCount++;
  1185 	for(ii=0;ii<count;++ii)
  1186 		{
  1187 		TRawEventHandler &handler = iEventHandlers[ii];
  1188 		if (handler.iEventHandler != NULL &&
  1189 			(!isPointerEvent ||
  1190 			 handler.iAdvancedPointersEnabled ||
  1191 			 aRawEvent.PointerNumber() == TWsPointer::PrimaryPointer()) &&
  1192 			handler.iEventHandler->OfferRawEvent(aRawEvent))
  1193 			{
  1194 			if (CClick::IsHandler())
  1195 				{
  1196 #ifdef LOG_WSERV_EVENTS
  1197 				RDebug::Print(_L("_WSEVENT_KEY: Send event %S for Key Click"), &RawEventName(aRawEvent));
  1198 #endif
  1199 				SendEventToKeyClick(aRawEvent);
  1200 				}
  1201 			eventHandled = ETrue;
  1202 #ifdef LOG_WSERV_EVENTS
  1203             RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent Event Consumed by ANIM.dll Handler No = %d Advanced Pointer Enabled = %d",ii,handler.iAdvancedPointersEnabled);
  1204 #endif
  1205 			break;
  1206 			}
  1207 		}
  1208 	if (--iEventHandlerCount == 0)
  1209 		{
  1210 		if (ERemovedEventHandlerWhileProcessingRawEvents & iBinaryFlags) // Handler was deleted while previous loop
  1211 			{ 
  1212 			iBinaryFlags &= ~ERemovedEventHandlerWhileProcessingRawEvents;
  1213 			for(ii=count-1;ii>=0;--ii)
  1214 				{
  1215 				if (iEventHandlers[ii].iEventHandler==NULL) iEventHandlers.Remove(ii);
  1216 				}
  1217 			}
  1218 		}
  1219 	if (eventHandled)
  1220 		{
  1221 #ifdef LOG_WSERV_EVENTS
  1222 		RDebug::Printf("_WSEVENT: Event is already handled by anim dll not by window server");
  1223 		// This is to determine when we press the power button which bring power dialog
  1224 		// whether it is a pointer event or key event
  1225 		// Also when we plugin the charging cable this is to determine whether it is a pointer event or key event
  1226 		RDebug::Print(_L("_WSEVENT: RawEvent Name = %S"), &RawEventName(aRawEvent));
  1227 #endif
  1228 		if (isPointerEvent)
  1229 			{
  1230 #ifdef LOG_WSERV_EVENTS
  1231 			RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent Pointer Number= %d  State = %x XY(%d,%d)",TWsPointer::iPointers[0].iNumber,TWsPointer::iPointers[0].iState,TWsPointer::iPointers[0].iPos.iX,TWsPointer::iPointers[0].iPos.iY);
  1232 		    RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent Pointer Number= %d  State = %x XY(%d,%d)",TWsPointer::iPointers[1].iNumber,TWsPointer::iPointers[1].iState,TWsPointer::iPointers[1].iPos.iX,TWsPointer::iPointers[1].iPos.iY);
  1233 #endif
  1234               //Prevention of the phone pointer event "dead lock". 
  1235               TPointerEvent::TType type;
  1236               TBool handled = ETrue;
  1237               GetPointerEvent(type, aRawEvent, handled);
  1238               switch(type)
  1239                   {
  1240                   case TPointerEvent::EButton1Down:
  1241                   case TPointerEvent::EButton2Down:
  1242                   case TPointerEvent::EButton3Down:
  1243                       TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateDown;
  1244                       break;
  1245                   case TPointerEvent::EButton1Up:
  1246                   case TPointerEvent::EButton2Up:
  1247                   case TPointerEvent::EButton3Up:
  1248                       TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateUp;
  1249                       break;
  1250                   case TPointerEvent::EOutOfRange:
  1251                       TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateOutOfRange;
  1252                       break;
  1253                       default:
  1254                       break;
  1255                   }
  1256 			}
  1257 		return;
  1258 		}
  1259 	
  1260 	switch(eventType)
  1261 		{
  1262 		case TRawEvent::ERedraw:
  1263 			CWsTop::RedrawScreens();
  1264 			break;
  1265 		case TRawEvent::ESwitchOn:
  1266 		case TRawEvent::ECaseOpen:
  1267 			{
  1268 			TInt event=EEventCaseOpened;
  1269 			CKeyboardRepeat::CancelRepeat(NULL);
  1270 			CWsPassword::SwitchOn();
  1271 			if (eventType==TRawEvent::ESwitchOn)
  1272 				{
  1273 				UserSvr::WsSwitchOnScreen();
  1274 				HAL::Set(HALData::EDisplayState,1);
  1275 				event=EEventSwitchOn;
  1276 				}
  1277 			ProcessEventQueue(iSwitchOnQueue, SendSwitchOnEvent, event, 0);
  1278 			break;
  1279 			}
  1280 		case TRawEvent::ESwitchOff:
  1281 		case TRawEvent::ECaseClose:
  1282 			{
  1283 			TBool switchOff=(eventType==TRawEvent::ESwitchOff);
  1284 			CWsTop::HandleSwitchOff(switchOff? EEventSwitchOff:EEventCaseClosed,switchOff);
  1285 			break;
  1286 			}
  1287 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP			
  1288 		case TRawEvent::ERestartSystem:
  1289 			{ /* restart event being handled */
  1290 			CWsTop::HandleSwitchOff(EEventRestartSystem,ETrue);
  1291 			break;
  1292 			}
  1293 #endif			
  1294 		case TRawEvent::EInactive:
  1295 #ifndef __WINS__
  1296 			CWsTop::WindowServer()->AnimationScheduler()->OnInactive();
  1297 #endif
  1298 			CKeyboardRepeat::CancelRepeat(NULL);
  1299 			break;
  1300 		case TRawEvent::EActive:
  1301 #ifndef __WINS__
  1302 			CWsTop::WindowServer()->AnimationScheduler()->OnActive();
  1303 #endif
  1304 			break;
  1305 		case TRawEvent::EKeyDown:
  1306 			{
  1307 #ifdef LOG_WSERV_EVENTS
  1308 			RDebug::Printf("_WSEVENT_KEY: TRawEvent::EKeyDown");
  1309 #endif
  1310 			_LIT(KWSERVDebugLogKeyDownArrival,"Key down arrives %d");
  1311 			CScreen* screen = CWsTop::Screen();
  1312 			WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
  1313 			if(CDebugBar* dbg = screen->DebugBar())
  1314 				dbg->OnKeyEvent();
  1315 			if (wsDebugLog)
  1316 				wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyDownArrival,aRawEvent.ScanCode());
  1317 			CKeyboardRepeat::KeyDown();
  1318 			TKeyData keyData;
  1319 			// Note iCaptureKeys is needed as dummy arg only. Key capture is
  1320 			// now handled in ProcessKeyPress().
  1321 			TBool translated=iKeyTranslator->TranslateKey(aRawEvent.ScanCode(), EFalse,*iCaptureKeys,keyData);
  1322 			ProcessModifierChanges();
  1323 			QueueKeyUpDown(aRawEvent);
  1324 			if (translated)
  1325 				{
  1326 				TKeyEvent keyEvent;
  1327 				keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
  1328 				keyEvent.iCode = keyData.iKeyCode;
  1329 				keyEvent.iModifiers = keyData.iModifiers;
  1330 				ProcessKeyPress(keyEvent, ETrue, 0);
  1331 				}
  1332 			}
  1333 			break;
  1334 		case TRawEvent::EKeyUp:
  1335 			{
  1336 #ifdef LOG_WSERV_EVENTS
  1337 			RDebug::Printf("_WSEVENT_KEY: TRawEvent::EKeyUp");
  1338 #endif
  1339 			_LIT(KWSERVDebugLogKeyUpArrival,"Key up arrives %d");
  1340 			CScreen* screen = CWsTop::Screen();
  1341 			WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
  1342 			if(CDebugBar* dbg = screen->DebugBar())
  1343 				dbg->OnKeyEvent();
  1344 			if (wsDebugLog)
  1345 				wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyUpArrival,aRawEvent.ScanCode());
  1346 			TKeyData keyData;
  1347 			CKeyboardRepeat::KeyUp(aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE);
  1348 			TBool translated=iKeyTranslator->TranslateKey(aRawEvent.ScanCode(), ETrue,*iCaptureKeys,keyData);
  1349 			ProcessModifierChanges();
  1350 			QueueKeyUpDown(aRawEvent);
  1351 			if (translated)
  1352 				{
  1353 				CKeyboardRepeat::CancelRepeat(NULL);
  1354 				TKeyEvent keyEvent;
  1355 				keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
  1356 				keyEvent.iCode = keyData.iKeyCode;
  1357 				keyEvent.iModifiers = keyData.iModifiers;
  1358 				ProcessKeyPress(keyEvent, EFalse, 0);
  1359 				}
  1360 			}
  1361 			break;
  1362 		case TRawEvent::EButton1Down:
  1363 		case TRawEvent::EButton2Down:
  1364 		case TRawEvent::EButton3Down:
  1365 		case TRawEvent::EPointerSwitchOn:
  1366 #ifndef __WINS__
  1367 			CWsTop::WindowServer()->AnimationScheduler()->OnActive();
  1368 #endif
  1369 			// fall through
  1370 		case TRawEvent::EButton1Up:
  1371 		case TRawEvent::EButton2Up:
  1372 		case TRawEvent::EButton3Up:
  1373 		case TRawEvent::EPointerMove:
  1374 		case TRawEvent::EPointer3DOutOfRange:
  1375 			#if defined(_DEBUG)
  1376 				WS_ASSERT_DEBUG(MousePress(aRawEvent,NULL), EWsPanicEventType);
  1377 			#else
  1378 				MousePress(aRawEvent,NULL);
  1379 			#endif
  1380 			break;
  1381 		case TRawEvent::EUpdateModifiers:
  1382 			iKeyTranslator->UpdateModifiers(aRawEvent.Modifiers());
  1383 			break;
  1384 		case TRawEvent::EKeyRepeat:
  1385  			{
  1386  			_LIT(KWSERVDebugLogRepeatingKeyArrival,"Repeating key arrives %d");
  1387  			if (wsDebugLog)
  1388  				wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogRepeatingKeyArrival,aRawEvent.ScanCode());
  1389 			TKeyEvent keyEvent;
  1390 			keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
  1391 			keyEvent.iCode = aRawEvent.ScanCode();
  1392  			keyEvent.iModifiers = iKeyTranslator->GetModifierState();
  1393 			ProcessKeyPress(keyEvent, EFalse, aRawEvent.Repeats());
  1394  			}
  1395  			break;
  1396 		default:
  1397 			break;
  1398 		}
  1399 #ifdef LOG_WSERV_EVENTS
  1400     RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent Number= %d  State = %x XY(%d,%d)",TWsPointer::iPointers[0].iNumber,TWsPointer::iPointers[0].iState,TWsPointer::iPointers[0].iPos.iX,TWsPointer::iPointers[0].iPos.iY);
  1401     RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::ProcessRawEvent Number= %d  State = %x XY(%d,%d)",TWsPointer::iPointers[1].iNumber,TWsPointer::iPointers[1].iState,TWsPointer::iPointers[1].iPos.iX,TWsPointer::iPointers[1].iPos.iY);
  1402 #endif
  1403 	}
  1404 
  1405 void TWindowServerEvent::ProcessKeyEvent(const TKeyEvent &aKeyEvent,TInt aRepeats)
  1406 	{
  1407 #ifdef LOG_WSERV_EVENTS
  1408     RDebug::Printf("_WSEVENT_KEY: TWindowServerEvent::ProcessKeyEvent, key code: %d, repeat: %d", aKeyEvent.iCode, aRepeats);
  1409 #endif
  1410 	TKeyData keyData;
  1411 	keyData.iModifiers=aKeyEvent.iModifiers;
  1412 	keyData.iApp=0;
  1413 	keyData.iHandle=0;
  1414 	keyData.iIsCaptureKey=EFalse;
  1415 	keyData.iKeyCode=aKeyEvent.iCode;
  1416 	if (CKeyboardRepeat::IsAreadyActive())
  1417 		{
  1418 		CKeyboardRepeat::CancelRepeat(NULL);
  1419 		}
  1420 	ProcessKeyPress(aKeyEvent, aRepeats == 0, aRepeats);
  1421 	}
  1422 
  1423 void TWindowServerEvent::AddCaptureKeyL(const TKeyCaptureRequest& aRequest)
  1424 	{
  1425 	iKeyEventRouter->AddCaptureKeyL(aRequest);
  1426 	}
  1427 
  1428 void TWindowServerEvent::UpdateCaptureKeyL(const TKeyCaptureRequest& aRequest)
  1429 	{
  1430 	iKeyEventRouter->UpdateCaptureKeyL(aRequest);
  1431 	}
  1432 
  1433 void TWindowServerEvent::CancelCaptureKey(TKeyCaptureType aType, TAny* aHandle)
  1434 	{
  1435 	iKeyEventRouter->CancelCaptureKey(aType, aHandle);
  1436 	}
  1437 
  1438 TInt TWindowServerEvent::GetModifierState()
  1439 	{
  1440 	return(iKeyTranslator->GetModifierState());
  1441 	}
  1442 
  1443 void TWindowServerEvent::SetModifierState(TEventModifier aModifier,TModifierState aState)
  1444 	{
  1445 	iKeyTranslator->SetModifierState(aModifier,aState);
  1446 	}
  1447 
  1448 TInt TWindowServerEvent::AddNotificationHandler(CAnim* aAnim, TUint32 aNotifications)
  1449 	{
  1450 	SNotificationHandler notif;
  1451 	notif.iAnim = aAnim;
  1452 	notif.iNotifications = aNotifications;
  1453 
  1454 	// update the entry if the anim is already in the array
  1455 	TInt count=iNotificationHandlers->Count();
  1456 	TInt ii;
  1457 	for(ii=0;ii<count;++ii)
  1458 		{
  1459 		if ((*iNotificationHandlers)[ii].iAnim==aAnim)
  1460 			{
  1461 			(*iNotificationHandlers)[ii]=notif;
  1462 			return KErrNone;
  1463 			}
  1464 		}
  1465 	
  1466 	// otherwise add it to the array
  1467 	TRAPD(err,iNotificationHandlers->AppendL(notif));
  1468 	return err;
  1469 	}
  1470 
  1471 void TWindowServerEvent::RemoveNotificationHandler(CAnim* aAnim)
  1472 	{
  1473 	TInt count=iNotificationHandlers->Count();
  1474 	TInt ii;
  1475 	for(ii=0;ii<count;++ii)
  1476 		{
  1477 		if ((*iNotificationHandlers)[ii].iAnim==aAnim)
  1478 			{
  1479 			iNotificationHandlers->Delete(ii);
  1480 			return;
  1481 			}
  1482 		}
  1483 	}
  1484 
  1485 void TWindowServerEvent::PublishNotification(const TWsEvent& aWsEvent)
  1486 	{
  1487 	TInt count=iNotificationHandlers->Count();
  1488 	TInt ii;
  1489 	for(ii=0;ii<count;++ii)
  1490 		{
  1491 		SNotificationHandler notif = (*iNotificationHandlers)[ii];
  1492 		switch (aWsEvent.Type())
  1493 			{
  1494 		case EEventDirectScreenAccessBegin:
  1495 		case EEventDirectScreenAccessEnd:
  1496 			if (notif.iNotifications & EDirectScreenAccess)
  1497 				{
  1498 				notif.iAnim->HandleNotification(aWsEvent);
  1499 				}
  1500 			break;
  1501 		case EEventHeartbeatTimerStateChange:
  1502 			if (notif.iNotifications & EHeartbeatTimer)
  1503 				{
  1504 				notif.iAnim->HandleNotification(aWsEvent);
  1505 				}
  1506 			break;
  1507 		case EEventScreenDeviceChanged:
  1508 			if (notif.iNotifications & EScreenDeviceChange)
  1509 				{
  1510 				notif.iAnim->HandleNotification(aWsEvent);
  1511 				}
  1512 			break;
  1513 		default:
  1514 			break;
  1515 			}
  1516 		}
  1517 
  1518 	}
  1519 
  1520 TBool TWindowServerEvent::DrawerCompareFunc(const TDrawerHandler& lhs, const TDrawerHandler& rhs)
  1521 	{
  1522 	return lhs.iDrawer == rhs.iDrawer;	
  1523 	}
  1524 
  1525 TInt TWindowServerEvent::RegisterDrawerHandler(CWsGraphicDrawer* aDrawer, TUint32 aEvents)
  1526 	{
  1527 	TInt idx = iDrawerHandlers->Find(TDrawerHandler(aDrawer, aEvents),
  1528 		TIdentityRelation<TDrawerHandler>(TWindowServerEvent::DrawerCompareFunc));
  1529 	if (idx != KErrNotFound)
  1530 		{
  1531 		// replace event mask for this drawer
  1532 		(*iDrawerHandlers)[idx].iEvents = aEvents;
  1533 		idx = KErrNone;
  1534 		}
  1535 	else
  1536 		idx = iDrawerHandlers->Append(TDrawerHandler(aDrawer,aEvents));
  1537 	
  1538 	return idx;
  1539 	}
  1540 
  1541 TInt TWindowServerEvent::UnregisterDrawerHandler(CWsGraphicDrawer* aDrawer)
  1542 	{
  1543 	TInt idx = iDrawerHandlers->Find(TDrawerHandler(aDrawer,0),
  1544 		TIdentityRelation<TDrawerHandler>(TWindowServerEvent::DrawerCompareFunc));
  1545 	if (idx == KErrNotFound)
  1546 		return idx;
  1547 	(*iDrawerHandlers)[idx].iDrawer = NULL;		//NotifyDrawer() will clean up the array
  1548 	return KErrNone;
  1549 	}
  1550 
  1551 TInt TWindowServerEvent::RegisterWsEventHandler(MWsEventHandler * aHandler, TUint32 aEvents)
  1552 	{
  1553 	TWsEventHandler handler(aHandler, aEvents);
  1554 	TInt idx = iWsEventHandlers.Find(handler, TIdentityRelation<TWsEventHandler>(TWsEventHandler::CompareHandler));
  1555 	if (idx < 0)
  1556 		{
  1557 		TInt err = iWsEventHandlers.Append(handler);
  1558 		return err;
  1559 		}
  1560 	else
  1561 		{
  1562 		iWsEventHandlers[idx].iEvents = aEvents;
  1563 		return KErrNone;
  1564 		}
  1565 	}
  1566 	
  1567 TInt TWindowServerEvent::UnregisterWsEventHandler(MWsEventHandler * aHandler)
  1568 	{
  1569 	TWsEventHandler handler(aHandler, 0);
  1570 	TInt idx = iWsEventHandlers.Find(handler, TIdentityRelation<TWsEventHandler>(TWsEventHandler::CompareHandler));
  1571 	if (idx < 0)
  1572 		return idx;
  1573 	iWsEventHandlers[idx].iEvents = NULL;	//NotifyDrawer() will clean up the array
  1574 	return KErrNone;
  1575 	}
  1576 
  1577 
  1578 void TWindowServerEvent::NotifyDrawer(const TWservCrEvent& aEvent)
  1579 	{
  1580 	TInt drawerCount = iDrawerHandlers->Count();
  1581 	for (TInt idx = 0; idx < drawerCount; idx++)
  1582 		{
  1583 		TDrawerHandler hd = (*iDrawerHandlers)[idx]; 
  1584 		if (!hd.iDrawer) 
  1585 			{                					//If the handler has been removed 
  1586 			iDrawerHandlers->Remove(idx);       //Remove from the array 
  1587 			drawerCount -= 1;   				//Update the counters 
  1588 			idx -= 1; 
  1589 			} 
  1590 		else 
  1591 			{ 
  1592 			if (hd.iEvents & aEvent.Type()) 
  1593 				{ 
  1594 			    hd.iDrawer->HandleEvent(aEvent); 
  1595 			    } 
  1596 			} 
  1597 		}
  1598 	
  1599 	TInt eventHandlerCount = iWsEventHandlers.Count();
  1600 	for (TInt idx = 0; idx < eventHandlerCount; ++idx)
  1601 		{
  1602 		TWsEventHandler* eh = &iWsEventHandlers[idx];
  1603 		if (!eh->iEvents)
  1604 			{								//If the handler has been removed
  1605 			iWsEventHandlers.Remove(idx);	//Remove from the array
  1606 			eventHandlerCount -= 1;			//Update the counters
  1607 			idx -= 1;
  1608 			}
  1609 		else
  1610 			{
  1611 			if (eh->iEvents & aEvent.Type())
  1612 				{
  1613 				eh->iHandler->DoHandleEvent(aEvent);
  1614 				}
  1615 			}
  1616 		}
  1617 	}
  1618 
  1619 void TWindowServerEvent::NotifyScreenDrawingEvent(const TRegion* aRegion)
  1620 	{
  1621 	if (aRegion && !aRegion->IsEmpty())
  1622 		{
  1623 		TWservCrEvent event(TWservCrEvent::EScreenDrawing,0,const_cast<TRegion*>(aRegion));
  1624 		NotifyDrawer(event);
  1625 		}
  1626 	}
  1627 
  1628 void TWindowServerEvent::NotifyScreenDrawingEvent(const TRect& aRect)
  1629 	{
  1630 	TRegionFix<1> reg(aRect);
  1631 	TWservCrEvent event(TWservCrEvent::EScreenDrawing,0,&reg);
  1632 	NotifyDrawer(event);
  1633 	}
  1634 
  1635 //
  1636 // CRawEventReceiver //
  1637 //
  1638 
  1639 CRawEventReceiver::CRawEventReceiver(TInt aPriority) : CActive(aPriority)
  1640 //
  1641 // Constructor
  1642 //
  1643 	{
  1644 	__DECLARE_NAME(_S("CRawEventReceiver"));
  1645 	}
  1646 
  1647 CRawEventReceiver::~CRawEventReceiver()
  1648 	{
  1649 	CActive::Cancel();
  1650 	}
  1651 
  1652 void CRawEventReceiver::ConstructL()
  1653 	{
  1654 	CActiveScheduler::Add(this);
  1655 	UserSvr::CaptureEventHook();
  1656 	Request();
  1657 	}
  1658 
  1659 void CRawEventReceiver::Request()
  1660 //
  1661 // Issue a request for the next event.
  1662 //
  1663 	{
  1664 	UserSvr::RequestEvent(iEventBuf,iStatus);
  1665 	SetActive();
  1666 	}
  1667 
  1668 void CRawEventReceiver::DoCancel()
  1669 //
  1670 // Cancel a pending event.
  1671 //
  1672 	{
  1673 	UserSvr::RequestEventCancel();
  1674 	}
  1675 
  1676 void CRawEventReceiver::RunL()
  1677 	{
  1678 #ifdef LOG_WSERV_EVENTS
  1679     RDebug::Printf("_WSEVENT_KEY: CRawEventReceiver::RunL Entry point for event receiver");
  1680 #endif
  1681 //__PROFILE_START(11);
  1682 	if (TWsPointer::PreProcessDriverEvent(iEventBuf.Event()
  1683 #if defined(__WINS__)
  1684 													,ETrue
  1685 #endif
  1686 														  ))
  1687 		TWindowServerEvent::ProcessRawEvent(iEventBuf.Event());
  1688 	Request();
  1689 //__PROFILE_END(11);
  1690 	}
  1691 
  1692 //
  1693 // TEventRequestQueue //
  1694 //
  1695 
  1696 TEventRequestQueue::TEventRequestQueue() : iQueue(_FOFF(TEventRequestItem,iQue))
  1697 	{}
  1698 
  1699 inline TSglQue<TEventRequestItem> &TEventRequestQueue::Queue()
  1700 	{return(iQueue);}
  1701 
  1702 TEventRequestItem *TEventRequestQueue::FindInEventRequestQueueList(const CWsWindowBase &aWindow)
  1703 //
  1704 // Return a pointer to the link in the queue for the window, or NULL if not in the queue
  1705 //
  1706 	{
  1707 	TSglQueIter<TEventRequestItem> iter(iQueue);
  1708 	TEventRequestItem *qPtr;
  1709 	while((qPtr=iter++)!=NULL)
  1710 		if (qPtr->iWindow==&aWindow)
  1711 			break;
  1712 	return(qPtr);
  1713 	}
  1714 
  1715 void TEventRequestQueue::AddToEventRequestListL(const CWsWindowBase &aWindow, TInt aParam, TEventControl aCircumstances)
  1716 //
  1717 // Add a link to the on event list
  1718 //
  1719 	{
  1720 	TEventRequestItem *item=FindInEventRequestQueueList(aWindow);
  1721 	if (!item)
  1722 		{
  1723 		item=new(ELeave) TEventRequestItem;
  1724 		item->iWindow= &aWindow;
  1725 		item->iParam=aParam;
  1726 		item->iCircumstances=aCircumstances;
  1727 		iQueue.AddFirst(*item);
  1728 		}
  1729 	item->iCircumstances=aCircumstances;
  1730 	item->iParam=aParam;	// Just update the parameter if already exists
  1731 	}
  1732 
  1733 void TEventRequestQueue::RemoveFromEventRequestListL(const CWsWindowBase &aWindow)
  1734 //
  1735 // Remove a link from the on event list
  1736 //
  1737 	{
  1738 	TEventRequestItem *qPtr=FindInEventRequestQueueList(aWindow);
  1739 	if (qPtr)
  1740 		{
  1741 		iQueue.Remove(*qPtr);
  1742 		delete qPtr;
  1743 		}
  1744 	}
  1745 
  1746 //
  1747 // Keyboard auto repeat class //
  1748 //
  1749 
  1750 CKeyboardRepeat::CKeyboardRepeat() : CTimer(EKeyRepeatPriority)
  1751 	{}
  1752 
  1753 void CKeyboardRepeat::NewL()
  1754 	{
  1755 	iThis=new(ELeave) CKeyboardRepeat();
  1756 	iThis->ConstructL();
  1757 	CActiveScheduler::Add(iThis);
  1758 	_LIT(KWSERVIniFileVarRepeatRollover,"REPEATROLLOVER");
  1759 	WsIniFile->FindVar(KWSERVIniFileVarRepeatRollover,iRepeatRollover);
  1760 	}
  1761 
  1762 void CKeyboardRepeat::Destroy()
  1763 	{
  1764 	delete iThis;
  1765 	}
  1766 
  1767 void CKeyboardRepeat::GetRepeatTime(TTimeIntervalMicroSeconds32 &aInitialTime, TTimeIntervalMicroSeconds32 &aTime)
  1768 	{
  1769 	aInitialTime=iInitialTime;
  1770 	aTime=iTime;
  1771 	}
  1772 
  1773 void CKeyboardRepeat::SetRepeatTime(const TTimeIntervalMicroSeconds32 &aInitialTime, const TTimeIntervalMicroSeconds32 &aTime)
  1774 	{
  1775 	iInitialTime=aInitialTime;
  1776 	iTime=aTime;
  1777 	}
  1778 
  1779 /**
  1780 Process timer events.
  1781 
  1782 Called when the key repeat timer expires, this function generates the
  1783 appropriate long key or repeated key event. If the timer was started for
  1784 normal key repeat or if the long key event was captured with the automatic
  1785 repeat option specified then the timer is restarted.
  1786 */
  1787 void CKeyboardRepeat::RunL()
  1788 	{
  1789 	User::ResetInactivityTime();
  1790 	WS_ASSERT_DEBUG(iRepeating != ERepeatNone, EWsPanicKeyRepeat);
  1791 	TBool timer=ETrue;
  1792 	if (iRepeating>=ERepeatLong)
  1793 		{
  1794 		// Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong
  1795 		WS_ASSERT_DEBUG(iLongCapture != NULL, EWsPanicKeyRepeat);
  1796 		if (iLongCapture)
  1797 			{
  1798 			iCurrentRepeat = iLongRepeat;
  1799 			timer = iLongCapture->iFlags & ELongCaptureRepeatEvents;
  1800 			iRepeating=ERepeatLongRepeated;
  1801 			}
  1802 		else
  1803 			{
  1804 			// Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong
  1805 			// Stop key repeat if this incorrect condition occurs
  1806 			timer=EFalse; 
  1807 			}		
  1808 		}
  1809 	if (timer)
  1810 		After(iTime);
  1811 	else
  1812 		iRepeating=ERepeatNone;
  1813 
  1814 	TWindowServerEvent::QueueKeyPress(iCurrentRepeat.iOutput, ETrue, 1);
  1815 	}
  1816 
  1817 /**
  1818 Start key repeat and long key press timer
  1819 
  1820 @param	aInputScanCode	Original scan code (before routing)
  1821 @param	aShortEvent		Short key event (routing plug-in output)
  1822 @param	aLongEvent		Pointer to long key event (routing plug-in output)
  1823 						or NULL if none.
  1824 
  1825 Note: When aLongEvent != NULL, iCurrentRepeat reflects the short key event
  1826 until the timer has expired. This is necessary to allow a delayed short key
  1827 event to be delivered by KeyUp(). CancelRepeat() must therefore examine
  1828 iCurrentRepeat or iLongRepeat according to the repeat type in iRepeat.
  1829 */
  1830 void CKeyboardRepeat::StartRepeat(TInt aInputScanCode, const TKeyEventRouterOutput& aShortEvent, const TKeyEventRouterOutput* aLongEvent)
  1831 	{
  1832 	TTimeIntervalMicroSeconds32 time;
  1833 	iCurrentRepeat.iInputScanCode = aInputScanCode;
  1834 	iCurrentRepeat.iOutput = aShortEvent;
  1835 
  1836 	if (aLongEvent)
  1837 		{
  1838 		iRepeating = ERepeatLong;
  1839 		iLongRepeat.iInputScanCode = aInputScanCode;
  1840 		iLongRepeat.iOutput = *aLongEvent;
  1841 		iLongCapture = static_cast<CWsCaptureLongKey*>(aLongEvent->iCaptureHandle);
  1842 		time = iLongCapture->iDelay;
  1843 		}
  1844 	else
  1845 		{
  1846 		iLongCapture = NULL;
  1847 		iRepeating=ERepeatNormal;
  1848 		time=iInitialTime;
  1849 		}
  1850 	iThis->After(time);
  1851 	}
  1852 
  1853 /**
  1854 Cancel key repeat processing
  1855 */
  1856 void CKeyboardRepeat::doCancelRepeat()
  1857 	{
  1858 	iRepeating=ERepeatNone;
  1859 	iThis->Cancel();
  1860 	}
  1861 
  1862 /**
  1863 Cancel any key repeat associated with the specified window group
  1864 
  1865 @param	aRepeatFocus	Destination window group or NULL for all
  1866 */
  1867 void CKeyboardRepeat::CancelRepeat(CWsWindowGroup *aRepeatFocus)
  1868 	{
  1869 	if (iRepeating != ERepeatNone)
  1870 		{
  1871 		if (aRepeatFocus == NULL ||
  1872 			(iRepeating == ERepeatNormal) && (aRepeatFocus == iCurrentRepeat.iOutput.iWindowGroup) ||
  1873 			(iRepeating >= ERepeatLong) && (aRepeatFocus == iLongRepeat.iOutput.iWindowGroup))
  1874 			{
  1875 			doCancelRepeat();
  1876 			iAlternateRepeatExists=EFalse;
  1877 			}
  1878 		}
  1879 	}
  1880 
  1881 /**
  1882 Cancel any key repeat associated with the specified capture handle
  1883 
  1884 @param	aCaptureHandle		Handle to capture request
  1885 @param	aLongCaptureFlag	ETrue for long key capture, EFalse for normal key
  1886 */
  1887 void CKeyboardRepeat::CancelRepeat(const TAny* aCaptureHandle, TBool aLongCaptureFlag)
  1888 	{
  1889 	if (aLongCaptureFlag)
  1890 		{
  1891 		// Cancel repeat for long capture key
  1892 		if (iRepeating >= ERepeatLong && aCaptureHandle == iLongRepeat.iOutput.iCaptureHandle)
  1893 			{
  1894 			doCancelRepeat();
  1895 			iAlternateRepeatExists=EFalse;
  1896 			}
  1897 		}
  1898 	else
  1899 		{
  1900 		// Cancel repeat for normal capture key
  1901 		if (iRepeating == ERepeatNormal && aCaptureHandle == iCurrentRepeat.iOutput.iCaptureHandle)
  1902 			{
  1903 			doCancelRepeat();
  1904 			iAlternateRepeatExists=EFalse;
  1905 			}
  1906 		}
  1907 	}
  1908 	
  1909 /**
  1910 Process a key down event during key repeat.
  1911 The current repeat data is saved for possible restoration after rollover.
  1912 */
  1913 void CKeyboardRepeat::KeyDown()
  1914 	{
  1915 	if (iRepeating!=ERepeatNone)
  1916 		{
  1917 		if (iRepeating==ERepeatNormal && iRepeatRollover>0) // 1 Allow key repeat rollover
  1918 			{
  1919 			iAlternateRepeat=iCurrentRepeat;
  1920 			iAlternateRepeatExists=ETrue;
  1921 			}
  1922 		doCancelRepeat();
  1923 		}
  1924 	}
  1925 
  1926 /**
  1927 Process a key up event during key repeat.
  1928 Send delayed short key event if necessary for long key event processing.
  1929 Switch to alternate repeat if rollover key was released.
  1930 
  1931 @param	aScanCode	Scan code
  1932 */
  1933 void CKeyboardRepeat::KeyUp(TInt aScanCode)
  1934 	{
  1935 	if (iAlternateRepeatExists && iAlternateRepeat.iInputScanCode == aScanCode)
  1936 		iAlternateRepeatExists=EFalse;
  1937 	if (iRepeating != ERepeatNone && iCurrentRepeat.iInputScanCode == aScanCode)
  1938 		{
  1939 		if (iRepeating==ERepeatLong)
  1940 			{
  1941 			// Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong			
  1942 			WS_ASSERT_DEBUG(iLongCapture != NULL, EWsPanicKeyRepeat);
  1943 			if (iLongCapture && !(iLongCapture->iFlags & ELongCaptureShortEventImmediately))
  1944 				{
  1945 				TWindowServerEvent::QueueKeyPress(iCurrentRepeat.iOutput, EFalse, 0);
  1946 				}
  1947 			}			
  1948 		if (iAlternateRepeatExists)
  1949 			{
  1950 			iAlternateRepeatExists=EFalse;
  1951 			iCurrentRepeat=iAlternateRepeat;
  1952 			iRepeating=ERepeatNormal;
  1953 			}
  1954 		else
  1955 			doCancelRepeat();
  1956 		}
  1957 	}
  1958 
  1959 //
  1960 // CWsHotKey //
  1961 //
  1962 
  1963 CWsHotKey::CWsHotKey(TInt aHotKeyType, TBool aIsDefault) : 
  1964 	iHotKeyType(aHotKeyType),
  1965 	iIsDefault(aIsDefault)
  1966 	{
  1967 	}
  1968 
  1969 CWsHotKey::~CWsHotKey()
  1970 	{
  1971 	delete iCaptureKey;
  1972 	}
  1973 
  1974 void CWsHotKey::ConstructLD(const TWsWinCmdCaptureKey &aCaptureKey)
  1975 	{
  1976 	CleanupStack::PushL(this);
  1977 	iCaptureKey=new(ELeave) CWsCaptureKey(NULL);
  1978 	iCaptureKey->ConstructL(aCaptureKey);
  1979 	CleanupStack::Pop();
  1980 	}
  1981 
  1982 void CWsHotKey::SetL(const TWsWinCmdCaptureKey &aCaptureKey)
  1983 	{
  1984 	iCaptureKey->SetL(aCaptureKey);
  1985 	}
  1986 
  1987 //
  1988 //CEventQueueRetry//
  1989 //
  1990 CEventQueueRetry* CEventQueueRetry::NewL()
  1991 	{
  1992 	CEventQueueRetry* self = new (ELeave) CEventQueueRetry;
  1993 	CleanupStack::PushL(self);
  1994 	self->ConstructL();
  1995 	CleanupStack::Pop(self);
  1996 	return self;
  1997 	}
  1998       
  1999 void CEventQueueRetry::ConstructL()
  2000 	{
  2001 	User::LeaveIfError(iTimer.CreateLocal());
  2002 	CActiveScheduler::Add(this);
  2003 	}
  2004  
  2005 CEventQueueRetry::CEventQueueRetry() : CActive(EPriorityStandard), iRetrySpinner(1)
  2006 	{
  2007 	
  2008 	}
  2009 
  2010 CEventQueueRetry::~CEventQueueRetry()
  2011 	{
  2012 	Cancel();
  2013 	iTimer.Close();
  2014 	iClientArray.Close();
  2015 	}
  2016 
  2017 void CEventQueueRetry::DoCancel()
  2018 	{
  2019 	iTimer.Cancel();
  2020 	iRetrySpinner = 0;
  2021 	iClientArray.Reset();
  2022 	}
  2023 
  2024 void CEventQueueRetry::RunL()
  2025 	{
  2026 	//some clients may be no longer interested in listening
  2027 	//so we need to refresh the client list each round of retry
  2028 	iClientArray.Reset();
  2029 	TInt err = iOwner->GetNotificationClients(iClientArray);
  2030 	if(err != KErrNone)
  2031 		{
  2032 		iClientArray.Reset(); //so the retry won't kick off
  2033 		}
  2034 	if(iClientArray.Count() > 0)
  2035 		{
  2036 		TBool eventOnAllQueues = ETrue;
  2037 		for(TInt i = 0; i < iClientArray.Count(); i++)
  2038 			{
  2039 			if(iClientArray[i]->RetryEvent(EEventDisplayChanged))
  2040 				{
  2041 				if(TWindowServerEvent::SendDisplayChangedEvents(iClientArray[i], iOwner->ScreenNumber(),
  2042 						iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
  2043 					{
  2044 					iClientArray[i]->RemoveRetryFlag(EEventDisplayChanged);
  2045 					}
  2046 				else
  2047 					{
  2048 					eventOnAllQueues = EFalse;
  2049 					}
  2050 				}
  2051 			}
  2052 		
  2053 		if(!eventOnAllQueues)
  2054 			{//another retry needed, but with increased time interval
  2055 			iRetrySpinner++;
  2056 			Retry(KRetryInitialDelay*iRetrySpinner);
  2057 			}
  2058 		}
  2059 	
  2060 	}
  2061 
  2062 void CEventQueueRetry::Retry(TInt aDelay)
  2063 	{
  2064 	//the retry might be infinite, with an increasing interval, as delivery of the event is garuanteed
  2065 	//if aDelay is greater than max TInt, it will be negative number so we reset it to KRetryInitialDelay
  2066 	if(aDelay < 0)
  2067 		{
  2068 		aDelay = KRetryInitialDelay;
  2069 		}
  2070 	iTimer.After(iStatus, aDelay);
  2071 	SetActive();
  2072 	}
  2073 
  2074 void CEventQueueRetry::Init(CScreen *aOwner)
  2075 	{
  2076 	iOwner = aOwner;
  2077 	iRetrySpinner = 0;
  2078 	
  2079 	}
  2080 
  2081 void CEventQueueRetry::CancelRetry()
  2082 	{
  2083 	Cancel();
  2084 	}