Update contrib.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Top level window server code
26 #include "windowgroup.h"
35 #include "advancedpointereventhelper.h"
36 #include "graphics/wsgraphicdrawerinternal.h"
37 #include "debughelper.h"
39 GLREF_D CDebugLogBase *wsDebugLog;
41 GLREF_C void StateDump();
42 GLREF_C void HeapDump();
44 _LIT(KDefaultKeyRouterPluginName, "keyrouter.dll");
45 _LIT(KWSERVIniFileVarKeyRouterPlugin, "KEYROUTERPLUGIN");
47 #define IMPOSSIBLE 0xFFFFFFFF
49 const TWsWinCmdCaptureKey ImpossibleKeyPress=
51 IMPOSSIBLE, // Impossible to hit key combination, used for disabling Hot Keys
55 const TWsWinCmdCaptureKey DefaultHotKeys[TWindowServerEvent::ENumHotKeys]={
57 EModifierFunc|EModifierCtrl|EModifierShift,
58 EModifierFunc|EModifierCtrl|EModifierShift,
61 EModifierFunc|EModifierCtrl|EModifierShift,
62 EModifierFunc|EModifierCtrl|EModifierShift,
64 { // Window server internal dump to log
65 EModifierFunc|EModifierCtrl|EModifierShift,
66 EModifierFunc|EModifierCtrl|EModifierShift,
69 EModifierFunc|EModifierCtrl|EModifierShift,
70 EModifierFunc|EModifierCtrl|EModifierShift,
72 { // Shutdown window server
74 EModifierFunc|EModifierCtrl|EModifierShift,
75 EModifierFunc|EModifierCtrl|EModifierShift,
78 IMPOSSIBLE, // Impossible to hit key combination, effectively disables shutdown key in release builds
83 EModifierFunc|EModifierCtrl|EModifierShift,
84 EModifierFunc|EModifierCtrl|EModifierShift,
106 { // Backlight toggle
109 EKeyBacklightToggle},
110 { // Screen Dimension Change
113 EKeyScreenDimension0},
117 EKeyScreenDimension1},
121 EKeyScreenDimension2},
125 EKeyScreenDimension3},
127 { // Display mode cycle
128 EModifierFunc|EModifierCtrl|EModifierShift,
129 EModifierFunc|EModifierCtrl|EModifierShift,
131 { // Orientation cycle
132 EModifierFunc|EModifierCtrl|EModifierShift,
133 EModifierFunc|EModifierCtrl|EModifierShift,
136 { // Display mode cycle
137 IMPOSSIBLE, // Impossible to hit key combination
140 { // Orientation cycle
141 IMPOSSIBLE, // Impossible to hit key combination
153 { // Cycle focus screen
154 EModifierFunc|EModifierCtrl|EModifierShift,
155 EModifierFunc|EModifierCtrl|EModifierShift,
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;
193 void TWindowServerEvent::DeleteHotKeys()
195 CWsHotKey *hotKey=iHotKeys;
198 CWsHotKey *next=hotKey->iNext;
205 void TWindowServerEvent::DeleteStatics()
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();
221 void TWindowServerEvent::InitStaticsL()
223 // Create the CEvent active object.
226 #if defined(__WINS__)
227 WS_ASSERT_DEBUG(TWindowServerEvent::ENumHotKeys==EHotKeyLastKeyType+1, EWsPanicUnknownCaptureKey);
229 iEventReceiver=new(ELeave) CRawEventReceiver(EEventPriority);
230 iEventReceiver->ConstructL();
231 iKeyTranslator=CKeyTranslator::New();
232 User::LeaveIfNull(iKeyTranslator);
234 // Change keyboard mapping according to information the HAL
236 if (HAL::Get(HALData::EKeyboardIndex,keyboardIndex)==KErrNone)
238 _LIT(KLitKeyDataDllName,"EKDATA.%02d");
239 TBuf<16> keyDataDllName;
240 keyDataDllName.Format(KLitKeyDataDllName,keyboardIndex);
241 iKeyTranslator->ChangeKeyData(keyDataDllName);
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();
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);
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);
270 _LIT(KLogLoadError, "Failed to load plugin '%S' (error %d)");
271 buf.Format(KLogLoadError, &pluginName, err);
274 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant, buf);
280 _LIT(KLoadError, "WServ: failed to load plugin '%S' (error %d)");
281 RDebug::Print(KLoadError, &pluginName, err);
286 // Create the key event router
287 typedef CKeyEventRouter* (*TCreateFunc)();
288 TCreateFunc newL = reinterpret_cast<TCreateFunc>(iKeyEventRouterLibrary.Lookup(1));
291 User::Leave(KErrNotFound);
293 iKeyEventRouter = (*newL)();
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);
304 void TWindowServerEvent::LinkHotKey(CWsHotKey *aWsHotKey)
306 aWsHotKey->SetLink(iHotKeys);
310 void TWindowServerEvent::ConstructDefaultHotKeyL(TInt aHotKey, const TWsWinCmdCaptureKey &aSystemKey)
312 CWsHotKey* hotKey=new(ELeave) CWsHotKey(aHotKey, ETrue);
313 // hotKey is pushed onto the cleanup stack in method ConstructLD.
314 hotKey->ConstructLD(aSystemKey);
318 CWsHotKey* TWindowServerEvent::ClearHotKeysL(TInt aHotKey)
320 if (aHotKey>ENumHotKeys)
322 User::Leave(KErrArgument);
324 CWsHotKey** pHotKey= &iHotKeys;
325 CWsHotKey* defaultHotKey=NULL;
328 TBool unlinked=EFalse;
329 if ((*pHotKey)->HotKeyType()==aHotKey)
331 CWsHotKey *free=*pHotKey;
332 if (free->IsDefault())
334 free->SetL(ImpossibleKeyPress);
339 *pHotKey=(*pHotKey)->iNext;
346 pHotKey=&(*pHotKey)->iNext;
349 return(defaultHotKey);
352 void TWindowServerEvent::ResetDefaultHotKeyL(TInt aHotKey)
354 if ((aHotKey<0) || (aHotKey>=ENumHotKeys))
356 User::Leave(KErrArgument);
358 CWsHotKey* defaultHotKey=ClearHotKeysL(aHotKey);
359 WS_ASSERT_DEBUG(defaultHotKey, EWsPanicDefaultHotKeyNotFound);
360 defaultHotKey->SetL(DefaultHotKeys[aHotKey]);
363 void TWindowServerEvent::SetHotKeyL(const TWsClCmdSetHotKey &aHotKey)
365 if (aHotKey.type>ENumHotKeys)
366 User::Leave(KErrArgument);
368 CWsHotKey *hotKey=new(ELeave) CWsHotKey(aHotKey.type, EFalse);
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);
380 void TWindowServerEvent::AddEventHandler(MEventHandler *aEventHandler, TBool aAdvancedPointersEnabled)
383 TRAPD(err,iEventHandlers.AppendL(TRawEventHandler(aEventHandler, aAdvancedPointersEnabled)));
384 WS_ASSERT_DEBUG(err==KErrNone, EWsPanicEventHandlerInconsistency);
386 iEventHandlers.AppendL(TRawEventHandler(aEventHandler, aAdvancedPointersEnabled)); //Shouldn't leave
388 #ifdef LOG_WSERV_EVENTS
389 RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::AddEventHandler Added handler = %d AdvancedPointerEnabled = %d", iEventHandlers.Count(),aAdvancedPointersEnabled);
393 void TWindowServerEvent::RemoveEventHandler(const MEventHandler *aEventHandler)
395 TInt count=iEventHandlers.Count();
397 for(ii=0;ii<count;++ii)
399 if (iEventHandlers[ii].iEventHandler==aEventHandler)
401 #ifdef LOG_WSERV_EVENTS
402 RDebug::Printf("_WSEVENT_POINTER: TWindowServerEvent::RemoveEventHandler Removed handler = %d",ii);
404 if (iEventHandlerCount>0)
406 iBinaryFlags |= ERemovedEventHandlerWhileProcessingRawEvents;
407 iEventHandlers[ii].iEventHandler=NULL; // replace the Handler with null to keep size of the array
411 iEventHandlers.Remove(ii);
418 void TWindowServerEvent::PotentialEventHandlerL(TInt aNum)
420 iPotentialEventHandlers+=aNum;
421 WS_ASSERT_DEBUG(iPotentialEventHandlers>=iEventHandlers.Count(), EWsPanicEventHandlerInconsistency);
422 TRAPD(err,iEventHandlers.Reserve(iPotentialEventHandlers));
428 else if (iPotentialEventHandlers==0)
429 iEventHandlers.Compress();
432 void SendSwitchOnEvent(TEventRequestItem *aQptr, TInt aEvent, TInt )
434 aQptr->iWindow->QueueEvent(aEvent);
437 /*void SendSwitchOffEvent(TEventRequestItem *aQptr, TInt , TInt )
439 aQptr->iWindow->QueueEvent(EEventSwitchOff);
442 void SendErrorMessage(TEventRequestItem *aQptr, TInt aCategory, TInt aError)
445 event.SetType(EEventErrorMessage);
446 event.SetHandle(aQptr->iWindow->ClientHandle());
447 event.ErrorMessage()->iErrorCategory=(TWsErrorMessage::TErrorCategory)aCategory;
448 event.ErrorMessage()->iError=aError;
450 aQptr->iWindow->EventQueue()->QueueEvent(event,EEventPriorityHigh);
453 void SendModifierChangedEvent(TEventRequestItem *aQptr, TInt aChanged, TInt )
455 TInt tmpChanged=aChanged&aQptr->iParam;
459 event.SetType(EEventModifiersChanged);
460 event.SetHandle(aQptr->iWindow->ClientHandle());
461 event.ModifiersChanged()->iChangedModifiers=tmpChanged;
462 event.ModifiersChanged()->iModifiers=TWindowServerEvent::GetStoredModifierState();
464 aQptr->iWindow->EventQueue()->QueueEvent(event,EEventPriorityHigh);
468 void TWindowServerEvent::ProcessEventQueue(TEventRequestQueue &aQueue, TSendEventFunc aFunc, TInt aParam1, TInt aParam2)
470 TSglQueIter<TEventRequestItem> iter(aQueue.Queue());
471 TEventRequestItem *qPtr;
472 CWsWindowGroup *focusWin=CWsTop::FocusWindowGroup();
473 while((qPtr=iter++)!=NULL)
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);
482 void TWindowServerEvent::NotifyOom()
486 TTimeIntervalSeconds interval;
487 TInt err=now.SecondsFrom(iPrevOomMessageTime,interval);
488 if (err!=KErrNone || interval.Int()<0 || interval.Int()>EOomEventSecondGap)
490 ProcessErrorMessages(TWsErrorMessage::EDrawingRegion,KErrNoMemory);
491 iPrevOomMessageTime=now;
495 TBool TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::TErrorCategory aCategory, TInt aError)
497 if (aError!=KErrNone)
499 ProcessEventQueue(iErrorMessageQueue, SendErrorMessage, aCategory, aError);
505 void TWindowServerEvent::ProcessModifierChanges()
507 TInt newState=iKeyTranslator->GetModifierState();
508 if (newState!=iModifierState)
510 TInt changed=iModifierState^newState;
511 iModifierState=newState;
512 ProcessEventQueue(iModifierChangedQueue, SendModifierChangedEvent, changed, 0);
516 TEventQueueWalkRet FindScreenDeviceChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
518 if (aEvent->Type()==EEventScreenDeviceChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
519 *(TUint *)aHandlePtr=0; // Indicates event found
520 return(EEventQueueWalkOk);
523 void TWindowServerEvent::SendScreenDeviceChangedEvents(CScreen* aScreen)
525 TSglQueIter<TEventRequestItem> iter(iScreenDeviceChangedQueue .Queue());
526 TEventRequestItem *qPtr;
527 while((qPtr=iter++)!=NULL)
528 SendScreenDeviceChangedEvent(qPtr->iWindow);
529 if(CClick::IsHandler())
531 TClickMakerData clickMakerData;
532 clickMakerData.screenDeviceMode=aScreen->ScreenSizeMode();
533 CClick::OtherEvent(EEventScreenDeviceChanged, &clickMakerData);
536 wsEvent.SetType(EEventScreenDeviceChanged);
537 TWindowServerEvent::PublishNotification(wsEvent);
538 TWservCrEvent crEvent(TWservCrEvent::EScreenSizeModeChanged,aScreen->ScreenSizeMode());
539 TWindowServerEvent::NotifyDrawer(crEvent);
542 void TWindowServerEvent::SendScreenDeviceChangedEvent(const CWsWindowBase *aWindow)
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);
551 TEventQueueWalkRet FindGroupChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
553 if (aEvent->Type()==EEventWindowGroupsChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
555 *(TUint *)aHandlePtr=0; // Indicates event found
557 return(EEventQueueWalkOk);
560 void TWindowServerEvent::SendGroupChangedEvents()
562 TSglQueIter<TEventRequestItem> iter(iGroupChangedQueue.Queue());
563 TEventRequestItem *qPtr;
564 while((qPtr=iter++)!=NULL)
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
572 queue->QueueEvent(handle, EEventWindowGroupsChanged);
577 TEventQueueWalkRet FindFocusChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
579 if (aEvent->Type()==EEventFocusGroupChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
581 *(TUint *)aHandlePtr=0; // Indicates event found
583 return(EEventQueueWalkOk);
586 void TWindowServerEvent::SendFocusChangedEvents()
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();
594 identifier=currentFocusWG->Identifier();
596 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EWindowGroupChanged,
597 screenNumber, reinterpret_cast<TAny*>(identifier)));
599 TSglQueIter<TEventRequestItem> iter(iFocusChangedQueue.Queue());
600 TEventRequestItem *qPtr;
601 while((qPtr=iter++)!=NULL)
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
609 queue->QueueEvent(handle, EEventFocusGroupChanged);
614 TEventQueueWalkRet FindGroupListChangedEvent(TAny *aHandlePtr, TWsEvent *aEvent)
616 if (aEvent->Type()==EEventWindowGroupListChanged && aEvent->Handle()==*(TUint *)aHandlePtr)
618 *(TUint *)aHandlePtr=0; // Indicates event found
620 return(EEventQueueWalkOk);
623 void TWindowServerEvent::SendGroupListChangedEvents()
625 TSglQueIter<TEventRequestItem> iter(iGroupListChangedQueue.Queue());
626 TEventRequestItem *qPtr;
627 while((qPtr=iter++)!=NULL)
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
635 queue->QueueEvent(handle, EEventWindowGroupListChanged);
640 TEventQueueWalkRet OverrideVisibilityChangedEvent(TAny *aNewEvent, TWsEvent *aOldEvent)
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())
648 aOldEvent->SetTimeNow();
649 aOldEvent->VisibilityChanged()->iFlags = newEvent->VisibilityChanged()->iFlags;
650 newEvent->SetHandle(NULL);
652 return EEventQueueWalkOk;
655 void TWindowServerEvent::SendVisibilityChangedEvents(CWsWindowBase* aWin, TUint aFlags)
657 CEventQueue *queue=aWin->EventQueue();
659 event.SetType(EEventWindowVisibilityChanged);
660 event.SetHandle(aWin->ClientHandle());
662 TWsVisibilityChangedEvent* visevent = event.VisibilityChanged();
663 visevent->iFlags = aFlags;
664 queue->WalkEventQueue(&OverrideVisibilityChangedEvent,&event);
665 if (event.Handle()!=NULL)
667 queue->QueueEvent(event);
671 TEventQueueWalkRet OverrideDisplayChangedEvent(TAny *aNewEvent, TWsEvent *aOldEvent)
673 TWsEvent* newEvent = static_cast<TWsEvent*>(aNewEvent);
674 if (aOldEvent->Type() == EEventDisplayChanged &&
675 aOldEvent->DisplayChanged()->iDisplayNumber == newEvent->DisplayChanged()->iDisplayNumber)
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
682 return EEventQueueWalkOk;
685 TBool TWindowServerEvent::SendDisplayChangedEvents(CWsClient* aWsClient, TInt aDisplayNumber, TInt aConfigurationChangeId, TInt aResolutionListChangeId)
687 CEventQueue *queue = aWsClient->EventQueue();
689 event.SetType(EEventDisplayChanged);
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
698 const CWsObject* obj=ptr->iObject;
699 if (obj && obj->Type()==WS_HANDLE_GROUP_WINDOW)
701 event.SetHandle(ptr->iHandle);
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)
714 return queue->QueueEvent(event);
719 void TWindowServerEvent::QueueKeyEvent(CWsWindowGroup *aWin, TWsEvent &aEvent, TWservEventPriorities aPriority)
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());
725 aWin->EventQueue()->QueueEvent(aEvent, aPriority);
729 Process a key press event.
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().
738 @param aKeyEvent Input key event
739 @param aCheckRepeat Check for key repeat and long key capture
740 @param aRepeats Repeat count
742 void TWindowServerEvent::ProcessKeyPress(const TKeyEvent& aKeyEvent, TBool aCheckRepeat, TInt aRepeats)
744 CWsWindowGroup* focusWin = CWsTop::FocusWindowGroup();
745 TUid focusAppUid = focusWin ? TUid::Uid(focusWin->Client()->SecureId().iId) : KNullUid;
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;
753 // RouteKey() must not fail. Check for leaves in case the plug-in
755 TRAPD(err, iKeyEventRouter->RouteKey(input, output));
756 WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
758 iKeyEventRouter->RouteKey(input, output);
761 WS_ASSERT_DEBUG(output.iResult == ERouted || output.iResult == ECaptured || output.iResult == EConsumed, EWsPanicKeyEventRouterBadResult);
763 if (output.iResult == EConsumed)
769 focusWin = static_cast<CWsWindowGroup*>(output.iWindowGroup);
771 WS_ASSERT_DEBUG((focusWin == NULL || focusWin->Type() == WS_HANDLE_GROUP_WINDOW) && (output.iResult != ERouted || focusWin == CWsTop::FocusWindowGroup()), EWsPanicKeyEventRouterBadWindowGroup);
773 // Ensure that short event is not marked with EModifierLongKey
774 output.iKeyEvent.iModifiers &= ~EModifierLongKey;
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)
783 output.iKeyEvent.iRepeats = aRepeats;
784 CClick::KeyEvent(EEventKey, output.iKeyEvent);
787 if (output.iResult == ECaptured)
789 if (output.iWindowGroup == NULL) // Captured by Wserv itself
791 _LIT(KWSERVDebugLogCapturedKey,"WSERV Captured Key");
792 CScreen* focusScreen=CWsTop::CurrentFocusScreen();
793 TInt screenNo=focusScreen->ScreenNumber();
796 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogCapturedKey);
797 CWsHotKey *hotKey=iHotKeys;
800 if (hotKey->KeyHandle() == reinterpret_cast<TInt>(output.iCaptureHandle))
802 switch(hotKey->HotKeyType())
804 case EHotKeyEnableLogging:
805 CWsTop::EnableLogging();
807 case EHotKeyDisableLogging:
808 CWsTop::DisableLogging();
810 case EHotKeyStateDump:
813 case EHotKeyHeapDump:
817 if (!CWsPassword::PasswordModeActive())
819 const TBool currentJustInTimeValue=User::JustInTime();
820 if (currentJustInTimeValue)
822 User::SetJustInTime(EFalse);
824 CWsTop::KillForegroundSession();
825 if (currentJustInTimeValue)
827 User::SetJustInTime(ETrue);
831 case EHotKeyShutDown:
834 case EHotKeyIncContrast:
835 focusScreen->IncContrast();
837 case EHotKeyDecContrast:
838 focusScreen->DecContrast();
841 CWsTop::HandleSwitchOff(EEventKeySwitchOff,ETrue);
843 case EHotKeyBacklightToggle:
846 if (!ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(screenNo,HALData::EBacklightState,state)))
847 ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,!state));
850 case EHotKeyBacklightOn:
851 ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,ETrue));
853 case EHotKeyBacklightOff:
854 ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(screenNo,HALData::EBacklightState,EFalse));
856 case EHotKeyScreenDimension0:
857 case EHotKeyScreenDimension1:
858 case EHotKeyScreenDimension2:
859 case EHotKeyScreenDimension3:
860 focusScreen->doSetScreenMode(hotKey->HotKeyType()-EHotKeyScreenDimension0);
862 case EHotKeyCycleDisplaySize:
863 focusScreen->CycleDisplaySize();
865 case EHotKeyCycleOrientation:
866 focusScreen->CycleOrientation();
868 case EHotKeyIncBrightness:
869 focusScreen->IncBrightness();
871 case EHotKeyDecBrightness:
872 focusScreen->DecBrightness();
874 case EHotKeyCycleFocusScreen:
875 CWsTop::SetCurrentFocusScreen((CWsTop::CurrentFocusScreen()->ScreenNumber()+1)%CWsTop::NumberOfScreens());
880 hotKey=hotKey->iNext;
882 WS_PANIC_ALWAYS(EWsPanicUnknownCaptureKey);
886 _LIT(KWSERVDebugLogKeyCapturedByApp,"Key captured by app %d");
888 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyCapturedByApp,focusWin->Identifier());
889 if (CWsPassword::PasswordModeActive() && focusWin!=CWsPassword::PasswordWindow()->WinGroup())
893 CWsCaptureLongKey* longCapture = NULL;
894 TKeyEventRouterOutput longOutput;
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;
902 TRAPD(err, iKeyEventRouter->RouteKey(input, longOutput));
903 WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
905 iKeyEventRouter->RouteKey(input, longOutput);
908 if (longOutput.iResult == ECaptured)
910 longCapture = static_cast<CWsCaptureLongKey*>(longOutput.iCaptureHandle);
912 // Mark long key events with EModifierLongKey so that applications
913 // can easily distinguish short and long events.
914 longOutput.iKeyEvent.iModifiers |= EModifierLongKey;
916 // Start timer to detect long key press
917 CKeyboardRepeat::StartRepeat(aKeyEvent.iScanCode, output, &longOutput);
919 else if (output.iResult != EConsumed && output.iKeyEvent.iModifiers & EModifierAutorepeatable)
921 // Start timer for key repeat
922 CKeyboardRepeat::StartRepeat(aKeyEvent.iScanCode, output, NULL);
926 // Queue the short event
927 if (!longCapture || longCapture->iFlags & ELongCaptureShortEventImmediately)
929 QueueKeyPress(output, EFalse, aRepeats);
934 Queue a key press event.
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().
940 @param aOutput Output key event from routing plug-in
941 @param aIsRepeat Event is due to key repeat
942 @param aRepeats Repeat count
944 void TWindowServerEvent::QueueKeyPress(const TKeyEventRouterOutput& aOutput, TBool aIsRepeat, TInt aRepeats)
946 if (aOutput.iResult == EConsumed)
948 // Don't deliver this key
953 TKeyEvent& keyEvent = *event.Key();
954 keyEvent = aOutput.iKeyEvent;
955 keyEvent.iRepeats = aRepeats;
957 CWsWindowGroup* focusWin = static_cast<CWsWindowGroup*>(aOutput.iWindowGroup);
958 WS_ASSERT_DEBUG(focusWin == NULL || focusWin->Type() == WS_HANDLE_GROUP_WINDOW, EWsPanicKeyEventRouterBadWindowGroup);
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)
964 event.SetType(EEventKey);
965 event.SetHandle(focusWin->ClientHandle());
968 CEventQueue* queue=focusWin->EventQueue();
970 const TWsEvent* prev=queue->PeekLastEvent();
971 if (prev != NULL && prev->Type() == EEventKey && prev->Key()->iRepeats > 0 && prev->Key()->iCode == keyEvent.iCode)
973 prev->Key()->iRepeats += aRepeats;
975 if (CClick::IsHandler())
976 CClick::KeyEvent(EEventKeyRepeat, *prev->Key());
980 if (CClick::IsHandler())
981 CClick::KeyEvent(EEventKeyRepeat,keyEvent);
983 QueueKeyEvent(focusWin, event, EEventPriorityLow);
988 Queue a key up/down event.
990 @param aRawEvent Raw event
992 void TWindowServerEvent::QueueKeyUpDown(const TRawEvent &aRawEvent)
994 #ifdef LOG_WSERV_EVENTS
995 RDebug::Print(_L("_WSEVENT_KEY: TWindowServerEvent::QueueKeyUpDown, Event Name: %S, Scan code: %d"), &RawEventName(aRawEvent), aRawEvent.ScanCode());
997 TEventCode type = aRawEvent.Type() == TRawEvent::EKeyUp ? EEventKeyUp : EEventKeyDown;
999 // Check for key up/down capture
1001 keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
1002 #if defined(__WINS__)
1003 keyEvent.iCode = __WINS_CHARCODE(aRawEvent.ScanCode());
1007 keyEvent.iModifiers = iModifierState;
1008 keyEvent.iRepeats = 0;
1010 CWsWindowGroup* focusWin = CWsTop::FocusWindowGroup();
1011 TUid focusAppUid = focusWin ? TUid::Uid(focusWin->Client()->SecureId().iId) : KNullUid;
1013 TKeyEventRouterInput input(ECaptureTypeKeyUpDown, keyEvent, focusWin, focusAppUid);
1014 TKeyEventRouterOutput output;
1016 TRAPD(err, iKeyEventRouter->RouteKey(input, output));
1017 WS_ASSERT_DEBUG(err == KErrNone, EWsPanicKeyEventRouterLeave);
1019 iKeyEventRouter->RouteKey(input, output);
1022 if (output.iResult == EConsumed)
1024 // Don't deliver this key. A key click is still generated for the
1026 if (CClick::IsHandler())
1028 CClick::KeyEvent(type, keyEvent);
1032 WS_ASSERT_DEBUG(output.iResult == ERouted || output.iResult == ECaptured, EWsPanicKeyEventRouterBadResult);
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())
1039 // Restore WINS character code
1040 output.iKeyEvent.iScanCode |= output.iKeyEvent.iCode;
1042 output.iKeyEvent.iCode = 0;
1045 output.iKeyEvent.iRepeats = 0;
1046 if (CClick::IsHandler())
1048 CClick::KeyEvent(type, output.iKeyEvent);
1052 *event.Key() = output.iKeyEvent;
1055 event.SetType(type);
1056 event.SetHandle(focusWin->ClientHandle());
1057 QueueKeyEvent(focusWin, event, EEventPriorityHigh);
1061 LOCAL_D void GetPointerEvent(TPointerEvent::TType& aType, const TRawEvent &aRawEvent, TBool& aHandled)
1064 switch(aRawEvent.Type())
1066 case TRawEvent::EButton1Down:
1067 aType=TPointerEvent::EButton1Down;
1069 case TRawEvent::EButton1Up:
1070 aType=TPointerEvent::EButton1Up;
1072 case TRawEvent::EButton2Down:
1073 aType=TPointerEvent::EButton2Down;
1075 case TRawEvent::EButton2Up:
1076 aType=TPointerEvent::EButton2Up;
1078 case TRawEvent::EButton3Down:
1079 aType=TPointerEvent::EButton3Down;
1081 case TRawEvent::EButton3Up:
1082 aType=TPointerEvent::EButton3Up;
1084 case TRawEvent::EPointerMove:
1085 aType=TPointerEvent::EMove;
1087 case TRawEvent::EPointerSwitchOn:
1088 aType=TPointerEvent::ESwitchOn;
1090 case TRawEvent::EPointer3DOutOfRange:
1091 aType=TPointerEvent::EOutOfRange;
1098 TBool TWindowServerEvent::MousePress(const TRawEvent &aRawEvent, const CWsWindowGroup *aGroupWin)
1100 //Return EFalse if known not to be a Mouse Event
1103 TBool handled=ETrue;
1104 TPointerEvent::TType type;
1105 GetPointerEvent(type, aRawEvent, handled);
1108 TPoint3D point3D(0,0,0);
1109 if (type != TPointerEvent::EOutOfRange)
1111 point3D = aRawEvent.Pos3D();
1114 TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, type,iKeyTranslator->GetModifierState(),point3D,aRawEvent.PointerNumber());
1115 TWsPointer::ProcessWsEvent(event, aGroupWin, ETrue);
1120 LOCAL_D void SendEventToKeyClick(const TRawEvent& aRawEvent)
1122 switch(aRawEvent.Type())
1124 case TRawEvent::EKeyDown:
1125 case TRawEvent::EKeyUp:
1129 keyEvent.iScanCode=aRawEvent.ScanCode();
1130 keyEvent.iModifiers=0;
1131 keyEvent.iRepeats=0;
1132 CClick::KeyEvent(EEventKey,keyEvent);
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:
1144 TBool handled=ETrue;
1145 TPointerEvent::TType type;
1146 GetPointerEvent(type, aRawEvent, handled);
1150 TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, type, 0, aRawEvent.Pos3D(), TPoint(), aRawEvent.PointerNumber());
1151 TAdvancedPointerEvent& pointerEvent = *event.Pointer();
1152 CClick::PointerEvent(pointerEvent.iPosition,pointerEvent);
1164 @param aRawEvent Raw event
1166 void TWindowServerEvent::ProcessRawEvent(const TRawEvent& aRawEvent)
1168 // Event has completed.
1171 TRawEvent::TType eventType = aRawEvent.Type();
1172 TBool isPointerEvent = TWsPointer::IsPointerEventType(eventType);
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);
1179 TWsPointer::UpdatePrimaryPointer(aRawEvent);
1181 TInt count=iEventHandlers.Count();
1183 TBool eventHandled = EFalse;
1184 iEventHandlerCount++;
1185 for(ii=0;ii<count;++ii)
1187 TRawEventHandler &handler = iEventHandlers[ii];
1188 if (handler.iEventHandler != NULL &&
1190 handler.iAdvancedPointersEnabled ||
1191 aRawEvent.PointerNumber() == TWsPointer::PrimaryPointer()) &&
1192 handler.iEventHandler->OfferRawEvent(aRawEvent))
1194 if (CClick::IsHandler())
1196 #ifdef LOG_WSERV_EVENTS
1197 RDebug::Print(_L("_WSEVENT_KEY: Send event %S for Key Click"), &RawEventName(aRawEvent));
1199 SendEventToKeyClick(aRawEvent);
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);
1208 if (--iEventHandlerCount == 0)
1210 if (ERemovedEventHandlerWhileProcessingRawEvents & iBinaryFlags) // Handler was deleted while previous loop
1212 iBinaryFlags &= ~ERemovedEventHandlerWhileProcessingRawEvents;
1213 for(ii=count-1;ii>=0;--ii)
1215 if (iEventHandlers[ii].iEventHandler==NULL) iEventHandlers.Remove(ii);
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));
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);
1234 //Prevention of the phone pointer event "dead lock".
1235 TPointerEvent::TType type;
1236 TBool handled = ETrue;
1237 GetPointerEvent(type, aRawEvent, handled);
1240 case TPointerEvent::EButton1Down:
1241 case TPointerEvent::EButton2Down:
1242 case TPointerEvent::EButton3Down:
1243 TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateDown;
1245 case TPointerEvent::EButton1Up:
1246 case TPointerEvent::EButton2Up:
1247 case TPointerEvent::EButton3Up:
1248 TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateUp;
1250 case TPointerEvent::EOutOfRange:
1251 TWsPointer::iPointers[aRawEvent.PointerNumber()].iState = TWsPointer::EPointerStateOutOfRange;
1262 case TRawEvent::ERedraw:
1263 CWsTop::RedrawScreens();
1265 case TRawEvent::ESwitchOn:
1266 case TRawEvent::ECaseOpen:
1268 TInt event=EEventCaseOpened;
1269 CKeyboardRepeat::CancelRepeat(NULL);
1270 CWsPassword::SwitchOn();
1271 if (eventType==TRawEvent::ESwitchOn)
1273 UserSvr::WsSwitchOnScreen();
1274 HAL::Set(HALData::EDisplayState,1);
1275 event=EEventSwitchOn;
1277 ProcessEventQueue(iSwitchOnQueue, SendSwitchOnEvent, event, 0);
1280 case TRawEvent::ESwitchOff:
1281 case TRawEvent::ECaseClose:
1283 TBool switchOff=(eventType==TRawEvent::ESwitchOff);
1284 CWsTop::HandleSwitchOff(switchOff? EEventSwitchOff:EEventCaseClosed,switchOff);
1287 #ifdef SYMBIAN_PROCESS_MONITORING_AND_STARTUP
1288 case TRawEvent::ERestartSystem:
1289 { /* restart event being handled */
1290 CWsTop::HandleSwitchOff(EEventRestartSystem,ETrue);
1294 case TRawEvent::EInactive:
1296 CWsTop::WindowServer()->AnimationScheduler()->OnInactive();
1298 CKeyboardRepeat::CancelRepeat(NULL);
1300 case TRawEvent::EActive:
1302 CWsTop::WindowServer()->AnimationScheduler()->OnActive();
1305 case TRawEvent::EKeyDown:
1307 #ifdef LOG_WSERV_EVENTS
1308 RDebug::Printf("_WSEVENT_KEY: TRawEvent::EKeyDown");
1310 _LIT(KWSERVDebugLogKeyDownArrival,"Key down arrives %d");
1311 CScreen* screen = CWsTop::Screen();
1312 WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
1313 if(CDebugBar* dbg = screen->DebugBar())
1316 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyDownArrival,aRawEvent.ScanCode());
1317 CKeyboardRepeat::KeyDown();
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);
1327 keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
1328 keyEvent.iCode = keyData.iKeyCode;
1329 keyEvent.iModifiers = keyData.iModifiers;
1330 ProcessKeyPress(keyEvent, ETrue, 0);
1334 case TRawEvent::EKeyUp:
1336 #ifdef LOG_WSERV_EVENTS
1337 RDebug::Printf("_WSEVENT_KEY: TRawEvent::EKeyUp");
1339 _LIT(KWSERVDebugLogKeyUpArrival,"Key up arrives %d");
1340 CScreen* screen = CWsTop::Screen();
1341 WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
1342 if(CDebugBar* dbg = screen->DebugBar())
1345 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogKeyUpArrival,aRawEvent.ScanCode());
1347 CKeyboardRepeat::KeyUp(aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE);
1348 TBool translated=iKeyTranslator->TranslateKey(aRawEvent.ScanCode(), ETrue,*iCaptureKeys,keyData);
1349 ProcessModifierChanges();
1350 QueueKeyUpDown(aRawEvent);
1353 CKeyboardRepeat::CancelRepeat(NULL);
1355 keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
1356 keyEvent.iCode = keyData.iKeyCode;
1357 keyEvent.iModifiers = keyData.iModifiers;
1358 ProcessKeyPress(keyEvent, EFalse, 0);
1362 case TRawEvent::EButton1Down:
1363 case TRawEvent::EButton2Down:
1364 case TRawEvent::EButton3Down:
1365 case TRawEvent::EPointerSwitchOn:
1367 CWsTop::WindowServer()->AnimationScheduler()->OnActive();
1370 case TRawEvent::EButton1Up:
1371 case TRawEvent::EButton2Up:
1372 case TRawEvent::EButton3Up:
1373 case TRawEvent::EPointerMove:
1374 case TRawEvent::EPointer3DOutOfRange:
1376 WS_ASSERT_DEBUG(MousePress(aRawEvent,NULL), EWsPanicEventType);
1378 MousePress(aRawEvent,NULL);
1381 case TRawEvent::EUpdateModifiers:
1382 iKeyTranslator->UpdateModifiers(aRawEvent.Modifiers());
1384 case TRawEvent::EKeyRepeat:
1386 _LIT(KWSERVDebugLogRepeatingKeyArrival,"Repeating key arrives %d");
1388 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVDebugLogRepeatingKeyArrival,aRawEvent.ScanCode());
1390 keyEvent.iScanCode = aRawEvent.ScanCode() __REMOVE_WINS_CHARCODE;
1391 keyEvent.iCode = aRawEvent.ScanCode();
1392 keyEvent.iModifiers = iKeyTranslator->GetModifierState();
1393 ProcessKeyPress(keyEvent, EFalse, aRawEvent.Repeats());
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);
1405 void TWindowServerEvent::ProcessKeyEvent(const TKeyEvent &aKeyEvent,TInt aRepeats)
1407 #ifdef LOG_WSERV_EVENTS
1408 RDebug::Printf("_WSEVENT_KEY: TWindowServerEvent::ProcessKeyEvent, key code: %d, repeat: %d", aKeyEvent.iCode, aRepeats);
1411 keyData.iModifiers=aKeyEvent.iModifiers;
1414 keyData.iIsCaptureKey=EFalse;
1415 keyData.iKeyCode=aKeyEvent.iCode;
1416 if (CKeyboardRepeat::IsAreadyActive())
1418 CKeyboardRepeat::CancelRepeat(NULL);
1420 ProcessKeyPress(aKeyEvent, aRepeats == 0, aRepeats);
1423 void TWindowServerEvent::AddCaptureKeyL(const TKeyCaptureRequest& aRequest)
1425 iKeyEventRouter->AddCaptureKeyL(aRequest);
1428 void TWindowServerEvent::UpdateCaptureKeyL(const TKeyCaptureRequest& aRequest)
1430 iKeyEventRouter->UpdateCaptureKeyL(aRequest);
1433 void TWindowServerEvent::CancelCaptureKey(TKeyCaptureType aType, TAny* aHandle)
1435 iKeyEventRouter->CancelCaptureKey(aType, aHandle);
1438 TInt TWindowServerEvent::GetModifierState()
1440 return(iKeyTranslator->GetModifierState());
1443 void TWindowServerEvent::SetModifierState(TEventModifier aModifier,TModifierState aState)
1445 iKeyTranslator->SetModifierState(aModifier,aState);
1448 TInt TWindowServerEvent::AddNotificationHandler(CAnim* aAnim, TUint32 aNotifications)
1450 SNotificationHandler notif;
1451 notif.iAnim = aAnim;
1452 notif.iNotifications = aNotifications;
1454 // update the entry if the anim is already in the array
1455 TInt count=iNotificationHandlers->Count();
1457 for(ii=0;ii<count;++ii)
1459 if ((*iNotificationHandlers)[ii].iAnim==aAnim)
1461 (*iNotificationHandlers)[ii]=notif;
1466 // otherwise add it to the array
1467 TRAPD(err,iNotificationHandlers->AppendL(notif));
1471 void TWindowServerEvent::RemoveNotificationHandler(CAnim* aAnim)
1473 TInt count=iNotificationHandlers->Count();
1475 for(ii=0;ii<count;++ii)
1477 if ((*iNotificationHandlers)[ii].iAnim==aAnim)
1479 iNotificationHandlers->Delete(ii);
1485 void TWindowServerEvent::PublishNotification(const TWsEvent& aWsEvent)
1487 TInt count=iNotificationHandlers->Count();
1489 for(ii=0;ii<count;++ii)
1491 SNotificationHandler notif = (*iNotificationHandlers)[ii];
1492 switch (aWsEvent.Type())
1494 case EEventDirectScreenAccessBegin:
1495 case EEventDirectScreenAccessEnd:
1496 if (notif.iNotifications & EDirectScreenAccess)
1498 notif.iAnim->HandleNotification(aWsEvent);
1501 case EEventHeartbeatTimerStateChange:
1502 if (notif.iNotifications & EHeartbeatTimer)
1504 notif.iAnim->HandleNotification(aWsEvent);
1507 case EEventScreenDeviceChanged:
1508 if (notif.iNotifications & EScreenDeviceChange)
1510 notif.iAnim->HandleNotification(aWsEvent);
1520 TBool TWindowServerEvent::DrawerCompareFunc(const TDrawerHandler& lhs, const TDrawerHandler& rhs)
1522 return lhs.iDrawer == rhs.iDrawer;
1525 TInt TWindowServerEvent::RegisterDrawerHandler(CWsGraphicDrawer* aDrawer, TUint32 aEvents)
1527 TInt idx = iDrawerHandlers->Find(TDrawerHandler(aDrawer, aEvents),
1528 TIdentityRelation<TDrawerHandler>(TWindowServerEvent::DrawerCompareFunc));
1529 if (idx != KErrNotFound)
1531 // replace event mask for this drawer
1532 (*iDrawerHandlers)[idx].iEvents = aEvents;
1536 idx = iDrawerHandlers->Append(TDrawerHandler(aDrawer,aEvents));
1541 TInt TWindowServerEvent::UnregisterDrawerHandler(CWsGraphicDrawer* aDrawer)
1543 TInt idx = iDrawerHandlers->Find(TDrawerHandler(aDrawer,0),
1544 TIdentityRelation<TDrawerHandler>(TWindowServerEvent::DrawerCompareFunc));
1545 if (idx == KErrNotFound)
1547 (*iDrawerHandlers)[idx].iDrawer = NULL; //NotifyDrawer() will clean up the array
1551 TInt TWindowServerEvent::RegisterWsEventHandler(MWsEventHandler * aHandler, TUint32 aEvents)
1553 TWsEventHandler handler(aHandler, aEvents);
1554 TInt idx = iWsEventHandlers.Find(handler, TIdentityRelation<TWsEventHandler>(TWsEventHandler::CompareHandler));
1557 TInt err = iWsEventHandlers.Append(handler);
1562 iWsEventHandlers[idx].iEvents = aEvents;
1567 TInt TWindowServerEvent::UnregisterWsEventHandler(MWsEventHandler * aHandler)
1569 TWsEventHandler handler(aHandler, 0);
1570 TInt idx = iWsEventHandlers.Find(handler, TIdentityRelation<TWsEventHandler>(TWsEventHandler::CompareHandler));
1573 iWsEventHandlers[idx].iEvents = NULL; //NotifyDrawer() will clean up the array
1578 void TWindowServerEvent::NotifyDrawer(const TWservCrEvent& aEvent)
1580 TInt drawerCount = iDrawerHandlers->Count();
1581 for (TInt idx = 0; idx < drawerCount; idx++)
1583 TDrawerHandler hd = (*iDrawerHandlers)[idx];
1585 { //If the handler has been removed
1586 iDrawerHandlers->Remove(idx); //Remove from the array
1587 drawerCount -= 1; //Update the counters
1592 if (hd.iEvents & aEvent.Type())
1594 hd.iDrawer->HandleEvent(aEvent);
1599 TInt eventHandlerCount = iWsEventHandlers.Count();
1600 for (TInt idx = 0; idx < eventHandlerCount; ++idx)
1602 TWsEventHandler* eh = &iWsEventHandlers[idx];
1604 { //If the handler has been removed
1605 iWsEventHandlers.Remove(idx); //Remove from the array
1606 eventHandlerCount -= 1; //Update the counters
1611 if (eh->iEvents & aEvent.Type())
1613 eh->iHandler->DoHandleEvent(aEvent);
1619 void TWindowServerEvent::NotifyScreenDrawingEvent(const TRegion* aRegion)
1621 if (aRegion && !aRegion->IsEmpty())
1623 TWservCrEvent event(TWservCrEvent::EScreenDrawing,0,const_cast<TRegion*>(aRegion));
1624 NotifyDrawer(event);
1628 void TWindowServerEvent::NotifyScreenDrawingEvent(const TRect& aRect)
1630 TRegionFix<1> reg(aRect);
1631 TWservCrEvent event(TWservCrEvent::EScreenDrawing,0,®);
1632 NotifyDrawer(event);
1636 // CRawEventReceiver //
1639 CRawEventReceiver::CRawEventReceiver(TInt aPriority) : CActive(aPriority)
1644 __DECLARE_NAME(_S("CRawEventReceiver"));
1647 CRawEventReceiver::~CRawEventReceiver()
1652 void CRawEventReceiver::ConstructL()
1654 CActiveScheduler::Add(this);
1655 UserSvr::CaptureEventHook();
1659 void CRawEventReceiver::Request()
1661 // Issue a request for the next event.
1664 UserSvr::RequestEvent(iEventBuf,iStatus);
1668 void CRawEventReceiver::DoCancel()
1670 // Cancel a pending event.
1673 UserSvr::RequestEventCancel();
1676 void CRawEventReceiver::RunL()
1678 #ifdef LOG_WSERV_EVENTS
1679 RDebug::Printf("_WSEVENT_KEY: CRawEventReceiver::RunL Entry point for event receiver");
1681 //__PROFILE_START(11);
1682 if (TWsPointer::PreProcessDriverEvent(iEventBuf.Event()
1683 #if defined(__WINS__)
1687 TWindowServerEvent::ProcessRawEvent(iEventBuf.Event());
1689 //__PROFILE_END(11);
1693 // TEventRequestQueue //
1696 TEventRequestQueue::TEventRequestQueue() : iQueue(_FOFF(TEventRequestItem,iQue))
1699 inline TSglQue<TEventRequestItem> &TEventRequestQueue::Queue()
1702 TEventRequestItem *TEventRequestQueue::FindInEventRequestQueueList(const CWsWindowBase &aWindow)
1704 // Return a pointer to the link in the queue for the window, or NULL if not in the queue
1707 TSglQueIter<TEventRequestItem> iter(iQueue);
1708 TEventRequestItem *qPtr;
1709 while((qPtr=iter++)!=NULL)
1710 if (qPtr->iWindow==&aWindow)
1715 void TEventRequestQueue::AddToEventRequestListL(const CWsWindowBase &aWindow, TInt aParam, TEventControl aCircumstances)
1717 // Add a link to the on event list
1720 TEventRequestItem *item=FindInEventRequestQueueList(aWindow);
1723 item=new(ELeave) TEventRequestItem;
1724 item->iWindow= &aWindow;
1725 item->iParam=aParam;
1726 item->iCircumstances=aCircumstances;
1727 iQueue.AddFirst(*item);
1729 item->iCircumstances=aCircumstances;
1730 item->iParam=aParam; // Just update the parameter if already exists
1733 void TEventRequestQueue::RemoveFromEventRequestListL(const CWsWindowBase &aWindow)
1735 // Remove a link from the on event list
1738 TEventRequestItem *qPtr=FindInEventRequestQueueList(aWindow);
1741 iQueue.Remove(*qPtr);
1747 // Keyboard auto repeat class //
1750 CKeyboardRepeat::CKeyboardRepeat() : CTimer(EKeyRepeatPriority)
1753 void CKeyboardRepeat::NewL()
1755 iThis=new(ELeave) CKeyboardRepeat();
1756 iThis->ConstructL();
1757 CActiveScheduler::Add(iThis);
1758 _LIT(KWSERVIniFileVarRepeatRollover,"REPEATROLLOVER");
1759 WsIniFile->FindVar(KWSERVIniFileVarRepeatRollover,iRepeatRollover);
1762 void CKeyboardRepeat::Destroy()
1767 void CKeyboardRepeat::GetRepeatTime(TTimeIntervalMicroSeconds32 &aInitialTime, TTimeIntervalMicroSeconds32 &aTime)
1769 aInitialTime=iInitialTime;
1773 void CKeyboardRepeat::SetRepeatTime(const TTimeIntervalMicroSeconds32 &aInitialTime, const TTimeIntervalMicroSeconds32 &aTime)
1775 iInitialTime=aInitialTime;
1780 Process timer events.
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.
1787 void CKeyboardRepeat::RunL()
1789 User::ResetInactivityTime();
1790 WS_ASSERT_DEBUG(iRepeating != ERepeatNone, EWsPanicKeyRepeat);
1792 if (iRepeating>=ERepeatLong)
1794 // Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong
1795 WS_ASSERT_DEBUG(iLongCapture != NULL, EWsPanicKeyRepeat);
1798 iCurrentRepeat = iLongRepeat;
1799 timer = iLongCapture->iFlags & ELongCaptureRepeatEvents;
1800 iRepeating=ERepeatLongRepeated;
1804 // Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong
1805 // Stop key repeat if this incorrect condition occurs
1812 iRepeating=ERepeatNone;
1814 TWindowServerEvent::QueueKeyPress(iCurrentRepeat.iOutput, ETrue, 1);
1818 Start key repeat and long key press timer
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)
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.
1830 void CKeyboardRepeat::StartRepeat(TInt aInputScanCode, const TKeyEventRouterOutput& aShortEvent, const TKeyEventRouterOutput* aLongEvent)
1832 TTimeIntervalMicroSeconds32 time;
1833 iCurrentRepeat.iInputScanCode = aInputScanCode;
1834 iCurrentRepeat.iOutput = aShortEvent;
1838 iRepeating = ERepeatLong;
1839 iLongRepeat.iInputScanCode = aInputScanCode;
1840 iLongRepeat.iOutput = *aLongEvent;
1841 iLongCapture = static_cast<CWsCaptureLongKey*>(aLongEvent->iCaptureHandle);
1842 time = iLongCapture->iDelay;
1846 iLongCapture = NULL;
1847 iRepeating=ERepeatNormal;
1854 Cancel key repeat processing
1856 void CKeyboardRepeat::doCancelRepeat()
1858 iRepeating=ERepeatNone;
1863 Cancel any key repeat associated with the specified window group
1865 @param aRepeatFocus Destination window group or NULL for all
1867 void CKeyboardRepeat::CancelRepeat(CWsWindowGroup *aRepeatFocus)
1869 if (iRepeating != ERepeatNone)
1871 if (aRepeatFocus == NULL ||
1872 (iRepeating == ERepeatNormal) && (aRepeatFocus == iCurrentRepeat.iOutput.iWindowGroup) ||
1873 (iRepeating >= ERepeatLong) && (aRepeatFocus == iLongRepeat.iOutput.iWindowGroup))
1876 iAlternateRepeatExists=EFalse;
1882 Cancel any key repeat associated with the specified capture handle
1884 @param aCaptureHandle Handle to capture request
1885 @param aLongCaptureFlag ETrue for long key capture, EFalse for normal key
1887 void CKeyboardRepeat::CancelRepeat(const TAny* aCaptureHandle, TBool aLongCaptureFlag)
1889 if (aLongCaptureFlag)
1891 // Cancel repeat for long capture key
1892 if (iRepeating >= ERepeatLong && aCaptureHandle == iLongRepeat.iOutput.iCaptureHandle)
1895 iAlternateRepeatExists=EFalse;
1900 // Cancel repeat for normal capture key
1901 if (iRepeating == ERepeatNormal && aCaptureHandle == iCurrentRepeat.iOutput.iCaptureHandle)
1904 iAlternateRepeatExists=EFalse;
1910 Process a key down event during key repeat.
1911 The current repeat data is saved for possible restoration after rollover.
1913 void CKeyboardRepeat::KeyDown()
1915 if (iRepeating!=ERepeatNone)
1917 if (iRepeating==ERepeatNormal && iRepeatRollover>0) // 1 Allow key repeat rollover
1919 iAlternateRepeat=iCurrentRepeat;
1920 iAlternateRepeatExists=ETrue;
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.
1931 @param aScanCode Scan code
1933 void CKeyboardRepeat::KeyUp(TInt aScanCode)
1935 if (iAlternateRepeatExists && iAlternateRepeat.iInputScanCode == aScanCode)
1936 iAlternateRepeatExists=EFalse;
1937 if (iRepeating != ERepeatNone && iCurrentRepeat.iInputScanCode == aScanCode)
1939 if (iRepeating==ERepeatLong)
1941 // Defensive programming - iLongCapture should never be NULL if iRepeating >= ERepeatLong
1942 WS_ASSERT_DEBUG(iLongCapture != NULL, EWsPanicKeyRepeat);
1943 if (iLongCapture && !(iLongCapture->iFlags & ELongCaptureShortEventImmediately))
1945 TWindowServerEvent::QueueKeyPress(iCurrentRepeat.iOutput, EFalse, 0);
1948 if (iAlternateRepeatExists)
1950 iAlternateRepeatExists=EFalse;
1951 iCurrentRepeat=iAlternateRepeat;
1952 iRepeating=ERepeatNormal;
1963 CWsHotKey::CWsHotKey(TInt aHotKeyType, TBool aIsDefault) :
1964 iHotKeyType(aHotKeyType),
1965 iIsDefault(aIsDefault)
1969 CWsHotKey::~CWsHotKey()
1974 void CWsHotKey::ConstructLD(const TWsWinCmdCaptureKey &aCaptureKey)
1976 CleanupStack::PushL(this);
1977 iCaptureKey=new(ELeave) CWsCaptureKey(NULL);
1978 iCaptureKey->ConstructL(aCaptureKey);
1979 CleanupStack::Pop();
1982 void CWsHotKey::SetL(const TWsWinCmdCaptureKey &aCaptureKey)
1984 iCaptureKey->SetL(aCaptureKey);
1988 //CEventQueueRetry//
1990 CEventQueueRetry* CEventQueueRetry::NewL()
1992 CEventQueueRetry* self = new (ELeave) CEventQueueRetry;
1993 CleanupStack::PushL(self);
1995 CleanupStack::Pop(self);
1999 void CEventQueueRetry::ConstructL()
2001 User::LeaveIfError(iTimer.CreateLocal());
2002 CActiveScheduler::Add(this);
2005 CEventQueueRetry::CEventQueueRetry() : CActive(EPriorityStandard), iRetrySpinner(1)
2010 CEventQueueRetry::~CEventQueueRetry()
2014 iClientArray.Close();
2017 void CEventQueueRetry::DoCancel()
2021 iClientArray.Reset();
2024 void CEventQueueRetry::RunL()
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);
2032 iClientArray.Reset(); //so the retry won't kick off
2034 if(iClientArray.Count() > 0)
2036 TBool eventOnAllQueues = ETrue;
2037 for(TInt i = 0; i < iClientArray.Count(); i++)
2039 if(iClientArray[i]->RetryEvent(EEventDisplayChanged))
2041 if(TWindowServerEvent::SendDisplayChangedEvents(iClientArray[i], iOwner->ScreenNumber(),
2042 iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
2044 iClientArray[i]->RemoveRetryFlag(EEventDisplayChanged);
2048 eventOnAllQueues = EFalse;
2053 if(!eventOnAllQueues)
2054 {//another retry needed, but with increased time interval
2056 Retry(KRetryInitialDelay*iRetrySpinner);
2062 void CEventQueueRetry::Retry(TInt aDelay)
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
2068 aDelay = KRetryInitialDelay;
2070 iTimer.After(iStatus, aDelay);
2074 void CEventQueueRetry::Init(CScreen *aOwner)
2081 void CEventQueueRetry::CancelRetry()