Update contrib.
1 // Copyright (c) 1995-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.
23 #include "windowgroup.h"
31 #include "advancedpointereventhelper.h"
32 #include "graphics/pointereventdata.h"
33 #include "debughelper.h"
35 TTimeIntervalMicroSeconds32 TWsPointer::iDoubleClickMaxInterval;
36 TInt TWsPointer::iDoubleClickMaxDistance;
37 CWsPointerCursor* TWsPointer::iCursorSprite;
38 TPointerCursorMode TWsPointer::iPointerCursorMode=EPointerCursorNormal;
39 TXYInputType TWsPointer::iXyInputType;
40 TBool TWsPointer::iTimerQueued;
41 TBool TWsPointer::iUpdateRequired;
42 CPeriodic* TWsPointer::iPeriodicTimer;
43 CWsRootWindow* TWsPointer::iRootWindow;
44 TInt TWsPointer::iMaxPointers;
45 TBool TWsPointer::iIs3DPointer;
46 RArray<TWsPointer> TWsPointer::iPointers;
47 TInt TWsPointer::iPrimaryPointer = TAdvancedPointerEvent::EDefaultPointerNumber;
48 TInt TWsPointer::iPreviousPrimaryPointer;
49 TInt TWsPointer::iEnterCloseProximityThreshold;
50 TInt TWsPointer::iExitCloseProximityThreshold;
51 TInt TWsPointer::iEnterHighPressureThreshold;
52 TInt TWsPointer::iExitHighPressureThreshold;
53 TBool CWsPointerBuffer::iSignalled=EFalse;
54 CWsPointerBuffer* CWsPointerBuffer::iCurrentBuffer=NULL;
55 CCirBuf<TPoint>* CWsPointerBuffer::iPointerBuffer=NULL;
56 TSglQue<CWsPointerBuffer> CWsPointerBuffer::iList(_FOFF(CWsPointerBuffer,iQue));
57 TInt TWsPointer::iYOffset;
59 TBool TWsPointer::iEmulatorRotatePointerCoords;
62 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_SwEvent,ECapabilitySwEvent);
64 void TWsPointer::InitStaticsL()
66 //This iYOffset setting is specific for capacitive touch screens, where user's finger is the pointer device.
67 //This is typically used so that the pointer event location is more inline with where the user perceives their
68 //finger to be on the screen (for example, due to refraction and the relatively large touch area of a finger).
70 _LIT( KWSERVIniFileVarYShifting, "YSHIFTING");
71 TBool fetchingSucceeded = WsIniFile->FindVar(KWSERVIniFileVarYShifting, iYOffset);
72 WS_ASSERT_ALWAYS(iYOffset>=0, EWsPanicInvalidPointerOffset);
73 if ( !fetchingSucceeded )
79 //An emulator may or may not deploy a renderchain or displaydriver that supports rotated drawing.
80 //On a real device target the coordinate system is always rotated together with wserv's screendevice.
81 _LIT( KWSERVIniFileVarEmulatorRotPointCoords, "EMULATOR_ROTATE_POINTER_COORDS");
82 iEmulatorRotatePointerCoords = WsIniFile->FindVar(KWSERVIniFileVarEmulatorRotPointCoords);
85 const CScreen* screen = CWsTop::Screen();
86 WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
87 iRootWindow = screen->RootWindow();
89 TMachineInfoV1Buf machineInfo;
90 UserHal::MachineInfo(machineInfo);
91 iXyInputType=machineInfo().iXYInputType;
93 // Read EPointerMaxPointers from HAL and check if reported value is consistent
95 // Even if HAL reports that device doesn't support any pointer, WSERV
96 // always has to support at least one pointer for compatibility reasons
97 // and for keeping state of pointer cursor.
98 if(HAL::Get(HALData::EPointerMaxPointers,iMaxPointers)!=KErrNone)
104 WS_ASSERT_ALWAYS(iMaxPointers >= 0 && iMaxPointers <= TAdvancedPointerEvent::EMaximumWServNumberOfPointers,
105 EWsPanicMaxPointersOutOfRange);
106 WS_ASSERT_ALWAYS(( XyInput() && iMaxPointers > 0) || (!XyInput() && iMaxPointers == 0),
107 EWsPanicMaxPointersInconsistent);
108 if (iMaxPointers == 0)
114 //** Log the number of pointers here i,e iMaxPointers
115 #ifdef LOG_WSERV_EVENTS
116 RDebug::Printf("_WSEVENT_POINTER: Number of pointers system supports %d", iMaxPointers);
119 // Does device support Z coordinate of the pointers?
120 if(HAL::Get(HALData::EPointer3D,iIs3DPointer)!=KErrNone)
122 iIs3DPointer=EFalse; // No API, then no 3D pointers
124 WS_ASSERT_ALWAYS(!iIs3DPointer || XyInput(), EWsPanicPointer3DInconsistent);
126 #ifdef LOG_WSERV_EVENTS
127 RDebug::Printf("_WSEVENT_POINTER: Z coordinate supported %d", iIs3DPointer);
130 // Initialize thresholds for EEnterCloseProximity, EExitCloseProximity,
131 // EEnterHighPressure and EExitHightPressure events.
132 if(HAL::Get(HALData::EPointer3DEnterCloseProximityThreshold,
133 iEnterCloseProximityThreshold) != KErrNone)
135 iEnterCloseProximityThreshold = KMaxTInt;
137 if(HAL::Get(HALData::EPointer3DExitCloseProximityThreshold,
138 iExitCloseProximityThreshold) != KErrNone)
140 iExitCloseProximityThreshold = KMinTInt;
142 if(HAL::Get(HALData::EPointer3DEnterHighPressureThreshold,
143 iEnterHighPressureThreshold) != KErrNone)
145 iEnterHighPressureThreshold = KMaxTInt;
147 if(HAL::Get(HALData::EPointer3DExitHighPressureThreshold,
148 iExitHighPressureThreshold) != KErrNone)
150 iExitHighPressureThreshold = KMinTInt;
153 iPointers = RArray<TWsPointer>(iMaxPointers);
154 TWsPointer emptyPointer;
155 emptyPointer.iRepeatTimer = NULL;
156 emptyPointer.Clear();
157 for (TInt ii = 0; ii < iMaxPointers; ii++)
159 emptyPointer.iNumber = ii;
160 User::LeaveIfError(iPointers.Append(emptyPointer));
161 RepeatTimer(ii) = CWsPointerTimer::NewL(iPointers[ii]);
164 iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
167 void TWsPointer::Clear()
169 iState = EPointerStateOutOfRange;
172 iPressureProximity = 0;
173 iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
174 iActualWinPointerIsOver = NULL;
176 iLastUnmatchedDown1 = NULL;
177 iLastUnmatchedDown2 = NULL;
178 iLastUnmatchedDown3 = NULL;
179 iLastUnmatchedEnterHighPressure = NULL;
180 iPrevClickWindow = NULL;
181 iInCloseProximity = EFalse;
182 iInHighPressure = EFalse;
183 CancelPointerRepeatEventRequest();
187 Turns off pointer cursor, deletes permenently pointer cursor update timer and pointer event repeat timer.
189 void TWsPointer::Stop()
191 SetPointerCursorMode(EPointerCursorNone);
192 UpdatePointerCursor();
193 delete iPeriodicTimer;
196 void TWsPointer::DeleteStatics()
198 for (TInt ii = 0; ii < iMaxPointers; ii++)
200 delete RepeatTimer(ii);
205 void TWsPointer::SetPointerCursorPos(TPoint aPos)
207 RestrictPos(aPos,EFalse);
208 iPointers[iPrimaryPointer].iPos=aPos;
209 iPointers[iPrimaryPointer].ReLogCurrentWindow();
210 UpdatePointerCursor();
213 void TWsPointer::SendEnterExitEvent(TEventCode aType)
216 && (iState != EPointerStateOutOfRange)
217 && (iNumber == iPrimaryPointer || iCurrentWindow->AdvancedPointersEnabled())
218 && !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
219 && !iCurrentWindow->ShutDownInProgress())
221 iCurrentWindow->QueueEvent(aType, iNumber);
225 void TWsPointer::SetCurrentWindow(const CWsWindow* aWin)
227 if (aWin!=iCurrentWindow)
229 SendEnterExitEvent(EEventPointerExit);
231 SendEnterExitEvent(EEventPointerEnter);
236 Relogs the current window (sets iCurrentWindow) for this pointer.
237 Retrieves pointer relative position in current window and pointer position in current
238 window's parent window.
240 void TWsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup* aForceInGroup)
244 SetCurrentWindow(iRootWindow->PointerWindow(iPos,&aPos,&aParentPos,iGrabWindow,
245 iActualWinPointerIsOver,aForceInGroup));
254 Relog all pointer's current windows when the window layout has changed.
256 void TWsPointer::ReLogPointersCurrentWindows()
258 for (TInt ii = 0; ii < iMaxPointers; ii++)
260 TWsPointer& pointer = iPointers[ii];
261 pointer.ReLogCurrentWindow();
266 Relog this pointer's current pointer window when the window layout has changed.
267 Works similarly to ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup* aForceInGroup),
268 but doesn't set relative positions.
270 void TWsPointer::ReLogCurrentWindow()
274 SetCurrentWindow(iRootWindow->PointerWindow(iPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
279 Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
280 is the current window
282 void TWsPointer::ReLogWindow(const CWsWindow* aWin)
284 for (TInt ii = 0; ii < iMaxPointers; ii++)
286 if (aWin == iPointers[ii].iCurrentWindow)
288 iPointers[ii].SendEnterExitEvent(EEventPointerEnter);
293 void TWsPointer::UnmatchedEventPurged(TPointerEvent::TType aPointerType, TUint aHandle)
295 if (aPointerType==TPointerEvent::EButton1Up)
297 if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
301 if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
303 CancelPointerRepeatEventRequest();
308 case TPointerEvent::EButton1Up:
309 iLastUnmatchedDown1=aHandle;
311 case TPointerEvent::EButton2Up:
312 iLastUnmatchedDown2=aHandle;
314 case TPointerEvent::EButton3Up:
315 iLastUnmatchedDown3=aHandle;
317 case TPointerEvent::EExitHighPressure:
318 iLastUnmatchedEnterHighPressure=aHandle;
324 void TWsPointer::WindowDisconnected(const CWsWindow* deletedWindow)
326 for (TInt pointerNum = 0; pointerNum < iMaxPointers; pointerNum++)
328 TWsPointer& pointer = iPointers[pointerNum];
329 if (pointer.iRepeatWindow==deletedWindow)
331 pointer.CancelPointerRepeatEventRequest();
333 if (pointer.iGrabWindow==deletedWindow)
335 pointer.iGrabWindow=NULL;
337 if (pointer.iCurrentWindow==deletedWindow)
339 pointer.ReLogCurrentWindow();
340 if (pointerNum == iPrimaryPointer)
342 UpdatePointerCursor();
349 Callback function pointer for up event remove event queue walk
351 TEventQueueWalkRet RemovePointerUpFunc(TAny* aHandle, TWsEvent* aEvent)
353 if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
355 return(EEventQueueWalkDeleteEvent);
357 return(EEventQueueWalkOk);
361 If the specified pointer is down claim grab in aWindow as though the down event had
362 gone to this window. Also send an up event to the window (if any) that would receive it if the
363 pointer was released now.
364 If no pointer is specifed do the above for the primary pointer.
366 @return KErrNone if successful,
367 KErrNotFound if pointernumber out of range,
368 KErrNotSupported if incorrect pointer grab claimed for window in emulation mode,
369 KErrPermissionDenied if trying to grab from a different window owner without the required capability.
371 TInt TWsPointer::ClaimGrab(const CWsWindow *aWindow,const TWsWinCmdGrabControl& aGrabControl)
373 TInt pointerNumber(aGrabControl.HasPointerNumber() ? aGrabControl.pointerNumber : iPrimaryPointer);
374 TBool advancedPointersEnabled(aWindow->AdvancedPointersEnabled());
375 TInt errNo(KErrNone);
377 if(!advancedPointersEnabled && aGrabControl.HasPointerNumber() && (TAdvancedPointerEvent::EDefaultPointerNumber!=pointerNumber))
379 // The window is in emulation mode, and cannot get events for anything other than the primary pointer
380 errNo=KErrNotSupported;
382 else if(advancedPointersEnabled && ((pointerNumber<0) || (pointerNumber>=iMaxPointers)))
384 // Unknown pointer number
387 else if(aGrabControl.HasPointerNumber() && (aWindow->WsOwner()!=iPointers[pointerNumber].iCurrentWindow->WsOwner())
388 && !KSecurityPolicy_SwEvent().CheckPolicy(aWindow->WsOwner()->ClientMessage(), __PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowBase::ClaimPointerGrab")))
390 // Trying to grab from a different window owner requires the relevant permission
391 // - Only for the multipointer API, RWindowBase::ClaimPointerGrab(TInt, TBool), which provides a pointer number
392 // - Doing so for the legacy non-multipointer API, RWindowBase::ClaimPointerGrab(TBool), would be a compatibility break
393 errNo=KErrPermissionDenied;
397 iPointers[pointerNumber].ClaimGrab(aWindow,aGrabControl.CheckFlags(TWsWinCmdGrabControl::ESendUpEvent));
404 If this pointer is down claim grab in aWindow as though the down event had
405 gone to this window. Also send an up event to the window (if any) that would receive it if the
406 pointer was released now.
408 void TWsPointer::ClaimGrab(const CWsWindow* aWindow,TBool aSendUpEvent)
410 TInt modState=TWindowServerEvent::GetModifierState();
412 TAdvancedPointerEvent& pointerEvent=*event.Pointer();
413 TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, TPointerEvent::EButton1Up,modState,TPoint3D(iPos.iX,iPos.iY,iPressureProximity),iNumber);
414 if (iState == EPointerStateDown)
416 if (iCurrentWindow!=aWindow)
420 ProcessEvent(event,NULL,EFalse);
422 else // If up event already in queue purge it
424 CEventQueue* eventQueue = iCurrentWindow->EventQueue();
427 TUint handle=iCurrentWindow->ClientHandle();
428 eventQueue->WalkEventQueue(&RemovePointerUpFunc,&handle);
431 iState = EPointerStateDown;
432 if (aWindow->HasPointerGrab())
436 ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
437 pointerEvent.iType=TPointerEvent::EDrag;
438 ProcessPointerEvent(event);
441 else if (iState == EPointerStateUp)
443 const CWsWindow *current=iCurrentWindow;
444 iCurrentWindow=aWindow;
445 WS_ASSERT_DEBUG(iGrabWindow==NULL, EWsPanicPointerClaimGrab);
446 iGrabWindow=aWindow; // Force the up event to be sent to aWindow
447 ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
448 ProcessPointerEvent(event);
450 iCurrentWindow=current;
455 @return ETrue if matching event has been purged, so current event should not be delivered,
458 TBool TWsPointer::CheckMatchingEventPurged(TPointerEvent::TType aType)
462 TUint lastUnmatchedDown;
463 case TPointerEvent::EButton1Up:
464 lastUnmatchedDown=iLastUnmatchedDown1;
465 iLastUnmatchedDown1=0;
466 iLastUnmatchedEnterHighPressure=0;
467 goto lastUnmatchedDownCheck;
468 case TPointerEvent::EButton2Up:
469 lastUnmatchedDown=iLastUnmatchedDown2;
470 iLastUnmatchedDown2=0;
471 goto lastUnmatchedDownCheck;
472 case TPointerEvent::EButton3Up:
473 lastUnmatchedDown=iLastUnmatchedDown3;
474 iLastUnmatchedDown3=0;
475 goto lastUnmatchedDownCheck;
476 case TPointerEvent::EExitHighPressure:
477 lastUnmatchedDown=iLastUnmatchedEnterHighPressure;
478 iLastUnmatchedEnterHighPressure=0;
479 lastUnmatchedDownCheck:
480 if (lastUnmatchedDown==iCurrentWindow->ClientHandle())
482 return ETrue; // Don't deliver the event as we've already thrown away matching event
484 default: //Should never get to default
490 TBool TWsPointer::QueuePointerEvent(const CWsWindow* aWindow, TWsEvent &aEvent)
492 if (aWindow->WsOwner() &&
493 (aWindow->AdvancedPointersEnabled() ||
494 TAdvancedPointerEventHelper::PointerNumber(aEvent) == iPrimaryPointer))
496 CEventQueue* queue=aWindow->EventQueue();
497 aEvent.SetHandle(aWindow->ClientHandle());
498 #ifdef LOG_WSERV_EVENTS
499 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent with AdvancedPointerEnabled");
501 if (aEvent.Handle()!=0)
503 if(!aWindow->AdvancedPointersEnabled())
505 // Re-assign from WServ primary pointer number, to EDefaultPointerNumber for Cone and other clients.
506 // This should not get confused with any real pointer of the same number due to the TWsEvents' window handle
508 TAdvancedPointerEventHelper::SetPointerNumber(aEvent,TAdvancedPointerEvent::EDefaultPointerNumber);
509 aEvent.Pointer()->iModifiers&=~EModifierAdvancedPointerEvent; // Clear the advanced pointer flag
511 if (queue->UpdateLastPointerEvent(aEvent))
515 TWservEventPriorities priority=EEventPriorityLow;
516 switch (aEvent.Pointer()->iType)
518 case TPointerEvent::EButton1Up:
519 case TPointerEvent::EButton2Up:
520 case TPointerEvent::EButton3Up:
521 case TPointerEvent::EExitHighPressure:
522 if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
524 #ifdef LOG_WSERV_EVENTS
525 RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 01");
529 if (queue->CheckRoom())
531 if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
533 #ifdef LOG_WSERV_EVENTS
534 RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 02");
539 /*Fall Through if an event was not purged*/
540 case TPointerEvent::EButton1Down:
541 case TPointerEvent::EButton2Down:
542 case TPointerEvent::EButton3Down:
543 case TPointerEvent::EEnterHighPressure:
544 case TPointerEvent::EOutOfRange:
545 priority=EEventPriorityHigh;
549 #ifdef LOG_WSERV_EVENTS
550 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent After adding event to clientqueue Event State %d ", iState);
552 queue->QueueEvent(aEvent,priority);
559 Moves the window group which contains this pointer's iCurrentWindow on top.
561 void TWsPointer::ProcessForegroundCheck()
563 CWsWindowGroup* group=((CWsTopClientWindow *)iCurrentWindow)->TopClientWindow()->Parent();
564 if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
566 group->SetOrdinalPosition(0);
571 Pointer Event Processing - stage 3 of 3:
572 - setting event's time
573 - auto foreground check
574 - double clicks detection
576 - add event to client's queue
577 - drag&drop capturing
579 void TWsPointer::ProcessPointerEvent(TWsEvent& aEvent)
581 if (iCurrentWindow && iCurrentWindow!=iRootWindow)
583 aEvent.SetType(EEventPointer);
585 TPointerEvent::TType type=aEvent.Pointer()->iType;
588 //TUint lastUnmatchedDown;
589 case TPointerEvent::EButton1Down:
590 ProcessForegroundCheck();
592 case TPointerEvent::EButton2Down:
593 case TPointerEvent::EButton3Down:
595 TPoint& pos=aEvent.Pointer()->iPosition;
596 if (iCurrentWindow==iPrevClickWindow &&
597 type==iPrevClickEventType &&
598 (Abs(pos.iX-iPrevClickPos.iX)+Abs(pos.iY-iPrevClickPos.iY))<iDoubleClickMaxDistance &&
599 aEvent.Time()<(iPrevClickTime+iDoubleClickMaxInterval))
601 aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
602 iPrevClickWindow=NULL; // Set to NULL to block a double double click
606 iPrevClickWindow=iCurrentWindow;
608 iPrevClickEventType=type;
610 iPrevClickTime=aEvent.Time();
618 if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
622 CancelPointerRepeatEventRequest();
624 if (QueuePointerEvent(iCurrentWindow, aEvent))
628 if (iCurrentWindow->DragDropCapture())
630 aEvent.SetType(EEventDragDrop);
631 QueuePointerEvent(iActualWinPointerIsOver, aEvent);
636 TInt PointerTimerCallBack(TAny *)
638 TWsPointer::TimerExpired();
642 void TWsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
644 CScreen* screen = iRootWindow->Screen();
645 WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
646 #if defined(__WINS__)
647 if (aWithinDrawableArea)
649 if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
651 return; //Not in the drawable area so user may be trying to click on facia button.
655 TRect validRect=screen->GetPointerCursorArea(screen->ScreenSizeMode());
656 if (aPos.iX<validRect.iTl.iX)
657 aPos.iX=validRect.iTl.iX;
658 else if (aPos.iX>=validRect.iBr.iX)
659 aPos.iX=validRect.iBr.iX-1;
660 if (aPos.iY<validRect.iTl.iY)
661 aPos.iY=validRect.iTl.iY;
662 else if (aPos.iY>=validRect.iBr.iY)
663 aPos.iY=validRect.iBr.iY-1;
666 #if defined(__WINS__)
667 TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
669 TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent)
672 TRawEvent::TType type=aRawEvent.Type();
674 if (!IsPointerEventType(type))
682 //** Log the type, pointer number, and its coordinates
683 #ifdef LOG_WSERV_EVENTS
684 RDebug::Printf("_WSEVENT_POINTER: Pointer number = %d RawEvent Type = %d Coordinates [%d, %d]",
685 aRawEvent.PointerNumber(), type, aRawEvent.Pos().iX, aRawEvent.Pos().iY);
688 // check correctness of aRawEvent.PointerNumber()
689 if (iMaxPointers > 1)
691 if (aRawEvent.PointerNumber() >= iMaxPointers)
698 aRawEvent.SetPointerNumber(0);
701 if (type != TRawEvent::EPointer3DOutOfRange)
703 // operations on coordinates are valid for all types except EPointer3DOutOfRange
704 TPoint xy=aRawEvent.Pos();
707 #if defined(__WINS__)
711 if (type==TRawEvent::EPointerMove)
713 xy+=iPointers[aRawEvent.PointerNumber()].iPos;
714 ShiftYCoordinate(xy.iY);
718 xy=iPointers[aRawEvent.PointerNumber()].iPos;
722 #if !defined(__WINS__)
723 TranslateCoordsOnRotation(xy);
725 if(iEmulatorRotatePointerCoords)
727 //emulators that support rotated drawing and touch may want to enable
728 //rotation of pointer events in wsini.
729 TranslateCoordsOnRotation(xy);
732 CScreen* screen=iRootWindow->Screen();
733 ShiftYCoordinate(xy.iY);
734 // Move the raw event position by shifting it by Origin and scale
735 xy=screen->PhysicalToLogical(xy);
738 aRawEvent.Set(type, xy.iX, xy.iY,
739 iIs3DPointer ? aRawEvent.Pos3D().iZ : 0);
742 #ifdef LOG_WSERV_EVENTS
743 RDebug::Printf("_WSEVENT_POINTER: Coordinates after Rotation and shift [%d, %d]",
744 aRawEvent.Pos().iX, aRawEvent.Pos().iY);
749 void TWsPointer::TranslateCoordsOnRotation(TPoint& aPoint)
751 CScreen* screen=iRootWindow->Screen();
752 TSize screenSize=screen->SizeInPixels()-TSize(1,1); //This is in the current rotation
753 switch (screen->Orientation())
755 case CFbsBitGc::EGraphicsOrientationRotated90:
756 aPoint.SetXY(aPoint.iY,screenSize.iHeight-aPoint.iX);
758 case CFbsBitGc::EGraphicsOrientationRotated180:
759 aPoint=-(aPoint-screenSize);
761 case CFbsBitGc::EGraphicsOrientationRotated270:
762 aPoint.SetXY(screenSize.iWidth-aPoint.iY,aPoint.iX);
764 default:; //To stop warning
769 * Validates events sent to the Window Server by its Client (Anim or Window Group).
770 * May overwrite aRawEvent's Z coordinate and/or pointer number if Client or digitizer driver
771 * doesn't support them.
772 * @param aRawEvent event to validate
773 * @param aAdvancedPointersEnabled ETrue if Client supports advanced pointer's data (Z coordiante
774 * and pointer number); EFalse otherwise.
775 * @return ETrue if aRawEvent should be processed by Window Server. EFalse if it should be ignored.
777 TBool TWsPointer::PreProcessClientEvent(TRawEvent &aRawEvent, TBool aAdvancedPointersEnabled)
779 TRawEvent::TType type=aRawEvent.Type();
781 if (!IsPointerEventType(type))
784 // validate pointer number
785 if (aAdvancedPointersEnabled)
787 // ignore event if both digitizer driver and Client support pointer numbers, but number is invalid
788 if (aRawEvent.PointerNumber() >= iMaxPointers)
795 // set to iPrimaryPointer if Client doesn't support pointer numbers
796 aRawEvent.SetPointerNumber(iPrimaryPointer);
799 // validate Z coordinate for all pointer events except EPointer3DOutOfRange
800 if (type != TRawEvent::EPointer3DOutOfRange)
804 // set Z to 0 for all events when the digitizer driver doesn't support Z coordinate
805 TPoint3D xyz=aRawEvent.Pos3D();
806 aRawEvent.Set(type, xyz.iX, xyz.iY, 0);
808 else if (!aAdvancedPointersEnabled)
810 // set Z to actual value if the digitizer driver does support Z coordinate but Client doesn't
811 TPoint3D xyz=aRawEvent.Pos3D();
812 aRawEvent.Set(type, xyz.iX, xyz.iY, iPointers[aRawEvent.PointerNumber()].iPressureProximity);
819 TBool TWsPointer::IsPointerEventType(TRawEvent::TType aType)
821 #if defined(__WINS__)
822 WS_ASSERT_DEBUG(TRawEvent::EPointerMove==1, EWsPanicRawEventsTypeChanged);
823 WS_ASSERT_DEBUG(TRawEvent::EPointerMove+1==TRawEvent::EPointerSwitchOn, EWsPanicRawEventsTypeChanged);
824 WS_ASSERT_DEBUG(TRawEvent::EPointerSwitchOn+8==TRawEvent::EButton1Down, EWsPanicRawEventsTypeChanged);
825 WS_ASSERT_DEBUG(TRawEvent::EButton1Down+5==TRawEvent::EButton3Up, EWsPanicRawEventsTypeChanged);
826 WS_ASSERT_DEBUG(TRawEvent::EButton3Up+6==TRawEvent::EPointer3DOutOfRange, EWsPanicRawEventsTypeChanged);
828 return (aType == TRawEvent::EPointerMove) ||
829 (aType == TRawEvent::EPointerSwitchOn) ||
830 (aType >= TRawEvent::EButton1Down && aType <= TRawEvent::EButton3Up) ||
831 (aType == TRawEvent::EPointer3DOutOfRange);
835 Pointer Event processing.
837 This method redirects pointer event processing to proper TWsPointer object.
839 void TWsPointer::ProcessWsEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
841 if(iPrimaryPointer!=iPreviousPrimaryPointer)
843 // The primary pointer may be updated while the TRawEvent is being processed, but this TRawEvent may be
844 // then be consumed by an anim.
845 // If it hasn't then we can leave any repeat request.
846 CancelPointerRepeatEventRequest(iPreviousPrimaryPointer);
849 if (aEvent.Pointer()->iType == TPointerEvent::EOutOfRange)
851 iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessOutOfRangeEvent(aEvent, aForceInGroup, aNatural);
855 iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessEvent(aEvent, aForceInGroup, aNatural);
860 Pointer Event Processing - stage 1 of 3:
861 - updating this pointer's state: coordinates and grabbing window
862 - updating coordinates of event
863 - updating event type to Drag
865 - generating events: EEnterCloseProximity, EExitCloseProximity, EEnterHighPressure, EExitHighPressure
867 void TWsPointer::ProcessEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
869 TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();
870 TPointerEvent::TType eventType=pointerEvent.iType;
872 if (iState == EPointerStateOutOfRange)
874 // new pointer comes in range, so clear information after previous one
878 // update coordinates
879 iPos=pointerEvent.iPosition;
880 iPressureProximity=TAdvancedPointerEventHelper::Z(aEvent);
882 if (eventType == TPointerEvent::EMove && !MovesAvailable() && iState != EPointerStateDown)
887 // re-log and update parent position
889 ReLogCurrentWindow(pointerEvent.iPosition,parPos,aForceInGroup);
890 pointerEvent.iParentPosition=parPos;
893 #ifdef LOG_WSERV_EVENTS
894 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::ProcessEvent Event to be sent to this window %U", reinterpret_cast<TUint32>(iCurrentWindow));
895 RDebug::Print(_L("_WSEVENT_POINTER: TWsPointer::ProcessEvent EventName %S and Event State %d "), &WsEventName(aEvent), iState);
900 case TPointerEvent::EButton1Down:
901 if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
903 iGrabWindow=iCurrentWindow;
905 if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
907 pointerEvent.iType=TPointerEvent::EMove;
908 ProcessEvent(aEvent, EFalse);
909 pointerEvent.iType=TPointerEvent::EButton1Down;
913 case EPointerStateOutOfRange:
914 iState = EPointerStateDown;
915 SendEnterExitEvent(EEventPointerEnter);
916 // intentional lack of break statement
917 case EPointerStateUp:
918 iState = EPointerStateDown;
919 ProcessEvent(aEvent, aNatural);
920 if (iPressureProximity >= iEnterHighPressureThreshold)
922 iInHighPressure = ETrue;
923 pointerEvent.iType = TPointerEvent::EEnterHighPressure;
924 ProcessEvent(aEvent, EFalse);
928 iInHighPressure = EFalse;
931 case EPointerStateDown:
932 if (iInHighPressure && iPressureProximity < iExitHighPressureThreshold)
934 iInHighPressure = EFalse;
935 eventType = TPointerEvent::EExitHighPressure;
937 else if (!iInHighPressure && iPressureProximity >= iEnterHighPressureThreshold)
939 iInHighPressure = ETrue;
940 eventType = TPointerEvent::EEnterHighPressure;
942 ProcessEvent(aEvent, aNatural);
946 case TPointerEvent::EButton1Up:
950 case EPointerStateDown:
951 iState = EPointerStateUp;
952 ProcessEvent(aEvent, aNatural);
953 if (iPressureProximity < iExitCloseProximityThreshold)
955 pointerEvent.iType = TPointerEvent::EExitCloseProximity;
956 iInCloseProximity = EFalse;
957 ProcessEvent(aEvent, EFalse);
961 iInCloseProximity = ETrue;
964 case EPointerStateOutOfRange:
965 iState = EPointerStateUp;
966 SendEnterExitEvent(EEventPointerEnter);
967 // intentional lack of break statement
968 case EPointerStateUp:
969 if (iInCloseProximity &&
970 iPressureProximity < iExitCloseProximityThreshold)
972 iInCloseProximity = EFalse;
973 pointerEvent.iType = TPointerEvent::EExitCloseProximity;
975 else if (!iInCloseProximity &&
976 iPressureProximity >= iEnterCloseProximityThreshold)
978 iInCloseProximity = ETrue;
979 pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
981 ProcessEvent(aEvent, aNatural);
985 case TPointerEvent::EMove:
988 case EPointerStateDown:
989 if (iInHighPressure &&
990 iPressureProximity < iExitHighPressureThreshold)
992 iInHighPressure = EFalse;
993 pointerEvent.iType = TPointerEvent::EExitHighPressure;
995 else if (!iInHighPressure &&
996 iPressureProximity >= iEnterHighPressureThreshold)
998 iInHighPressure = ETrue;
999 pointerEvent.iType = TPointerEvent::EEnterHighPressure;
1003 pointerEvent.iType = TPointerEvent::EDrag;
1006 case EPointerStateUp:
1007 if (iInCloseProximity &&
1008 iPressureProximity < iExitCloseProximityThreshold)
1010 iInCloseProximity = EFalse;
1011 pointerEvent.iType = TPointerEvent::EExitCloseProximity;
1013 else if (!iInCloseProximity &&
1014 iPressureProximity >= iEnterCloseProximityThreshold)
1016 iInCloseProximity = ETrue;
1017 pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
1020 case EPointerStateOutOfRange:
1021 iState = EPointerStateUp;
1022 SendEnterExitEvent(EEventPointerEnter);
1023 if (iPressureProximity >= iEnterCloseProximityThreshold)
1025 iInCloseProximity = ETrue;
1026 pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
1030 ProcessEvent(aEvent, aNatural);
1033 ProcessEvent(aEvent, aNatural);
1039 Processes OutOfRange events:
1040 - injects event to key click plugin
1041 - directs event to the last current window
1044 void TWsPointer::ProcessOutOfRangeEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup, TBool aNatural)
1046 if (iState != EPointerStateOutOfRange)
1048 #ifdef LOG_WSERV_EVENTS
1049 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::ProcessOutOfRangeEvent Pointer Number = %d, iState =%d ", iNumber, iState);
1052 // OutOfRange event generated by driver doesn't contain correct coordinates,
1053 // we update them from last state in order to deliver event to the proper window.
1054 SendEnterExitEvent(EEventPointerExit);
1056 iState = EPointerStateOutOfRange;
1058 TAdvancedPointerEventHelper::SetPointerNumber(aEvent, iNumber);
1059 TAdvancedPointerEventHelper::SetZ(aEvent, iPressureProximity);
1061 TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();
1062 iCurrentWindow=iRootWindow->PointerWindow(iPos,&pointerEvent.iPosition,&pointerEvent.iParentPosition,iGrabWindow,
1063 iActualWinPointerIsOver,aForceInGroup);
1064 ProcessEvent(aEvent, aNatural);
1068 void TWsPointer::NotifyCClick(TAdvancedPointerEvent& aPointerEvent)
1070 if (CClick::IsHandler())
1072 CClick::PointerEvent(iPos,aPointerEvent);
1073 TPointerEventData params;
1075 params.iCurrentPos=iPos;
1076 TAdvancedPointerEventHelper::Copy(aPointerEvent, params.iPointerEvent);
1077 params.iClientHandle=iCurrentWindow->ClientHandle();
1078 params.iWindowOrigin=iCurrentWindow->Origin();
1079 CWsWindowGroup* groupWin=iCurrentWindow->WinGroup();
1080 params.iWindowGroupId=groupWin ? groupWin->Identifier() : 0;
1081 params.iSource=TPointerEventData::EUnspecified;
1082 CClick::OtherEvent(EEventPointer,¶ms);
1087 Pointer Event Processing - stage 2 of 3:
1088 - injecting events to key click plugin
1090 - injection to event buffer
1091 - injection to keyboard emulator
1092 - clearing iCurrentWindow if going up and !MovesAvailable()
1093 - updating pointer cursor
1095 void TWsPointer::ProcessEvent(TWsEvent& aEvent, TBool aNatural)
1099 NotifyCClick(*aEvent.Pointer());
1102 TUint filter=iCurrentWindow->PointerFilter();
1103 TPointerEvent::TType type=aEvent.Pointer()->iType;
1104 if ((type!=TPointerEvent::EMove || !(filter&EPointerFilterMove)) &&
1105 (type!=TPointerEvent::EDrag || !(filter&EPointerFilterDrag)))
1107 if (iNumber == iPrimaryPointer)
1109 TPoint pos=aEvent.Pointer()->iPosition;
1110 if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
1112 CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
1114 else if ((type==TPointerEvent::EEnterCloseProximity || type==TPointerEvent::EExitCloseProximity ||
1115 type==TPointerEvent::EEnterHighPressure || type==TPointerEvent::EExitHighPressure) &&
1116 iCurrentWindow->UsingPointerBuffer())
1118 CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
1119 ProcessPointerEvent(aEvent);
1121 else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
1123 #ifdef LOG_WSERV_EVENTS
1124 RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for primary pointer");
1126 ProcessPointerEvent(aEvent);
1129 else if (!iCurrentWindow->UsingPointerBuffer() || (type != TPointerEvent::EMove && type != TPointerEvent::EDrag))
1131 #ifdef LOG_WSERV_EVENTS
1132 RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for non primary pointer");
1134 ProcessPointerEvent(aEvent);
1137 if (!MovesAvailable() && (type==TPointerEvent::EButton1Up ||
1138 type==TPointerEvent::ESwitchOn ||
1139 type==TPointerEvent::EOutOfRange))
1141 iCurrentWindow=NULL;
1143 if (iNumber == iPrimaryPointer)
1145 PointerCursorUpdateCheck();
1149 void TWsPointer::TimerExpired()
1151 WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
1152 if (iUpdateRequired)
1154 UpdatePointerCursor();
1155 iUpdateRequired=EFalse;
1159 iTimerQueued=EFalse;
1160 iPeriodicTimer->Cancel();
1164 void TWsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
1166 aTime=iDoubleClickMaxInterval;
1167 aDistance=iDoubleClickMaxDistance;
1170 void TWsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
1172 iDoubleClickMaxInterval=aTime;
1173 iDoubleClickMaxDistance=aDistance;
1176 void TWsPointer::PointerCursorUpdateCheck()
1178 CWsPointerCursor* sprite=CalculatePointerCursor();
1179 if (iCursorSprite || sprite) // If there either was, or is a pointer cursor we need an update
1183 UpdatePointerCursorTo(sprite);
1184 iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1185 TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1186 TCallBack(PointerTimerCallBack,NULL));
1191 iUpdateRequired=ETrue;
1196 void TWsPointer::UpdatePointerCursor()
1198 //__PROFILE_START(3);
1199 CWsPointerCursor* sprite=iPointers[iPrimaryPointer].CalculatePointerCursor();
1200 UpdatePointerCursorTo(sprite);
1204 void TWsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
1206 if (iCursorSprite!=aNewCursor)
1210 iCursorSprite->Deactivate();
1212 iCursorSprite=aNewCursor;
1215 iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
1216 iCursorSprite->Activate();
1219 else if (iCursorSprite)
1221 iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
1225 CWsPointerCursor* TWsPointer::CalculatePointerCursor()
1227 CWsPointerCursor* sprite=NULL;
1228 if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
1230 const CWsWindowBase* window=iCurrentWindow;
1232 sprite=window->PointerCursor();
1233 if (window->WinType()!=EWinTypeClient)
1237 window=window->BaseParent();
1240 if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
1242 sprite=CWsClient::DefaultSystemPointerCursor();
1248 Callback function for event queue walk
1250 TEventQueueWalkRet PointerRepeatPurgeFunc(TAny* aReqPtrNum, TWsEvent* aQueueEvent)
1252 return(TWsPointer::PointerRepeatPurgeCheck(aQueueEvent, reinterpret_cast<TUint>(aReqPtrNum)));
1255 TBool TWsPointer::PointerEventRepeatCheck(const TWsEvent* aEvent, TUint32 aHandle)
1257 // Return ETrue if this pointer event is consumed by the pointer repeat
1260 // Must be a pointer event type in order to get the pointer number,
1261 // which is needed to check the repeat window.
1262 WS_ASSERT_DEBUG(aEvent->Type()==EEventPointer,EWsPanicEventType);
1264 const TAdvancedPointerEvent* pntEvent=aEvent->Pointer();
1265 if ( TAdvancedPointerEventHelper::PointerNumber(*aEvent)==iNumber &&
1266 aHandle==iRepeatWindow->ClientHandle())
1268 switch(pntEvent->iType)
1270 case TPointerEvent::EDrag: // deliberate drop-through
1271 case TPointerEvent::EMove: // deliberate drop-through
1272 case TPointerEvent::EEnterCloseProximity: // deliberate drop-through
1273 case TPointerEvent::EExitCloseProximity: // deliberate drop-through
1274 case TPointerEvent::EEnterHighPressure: // deliberate drop-through
1275 case TPointerEvent::EExitHighPressure:
1277 if(iRepeatRect.Contains(pntEvent->iPosition))
1284 // do nothing and drop through
1291 TEventQueueWalkRet TWsPointer::PointerRepeatPurgeCheck(TWsEvent* aQueueEvent, TUint8 aReqPtrNum)
1293 // Return value is "WalkOK", unless a repeated event is found that needs to be deleted.
1294 TEventQueueWalkRet eventQueueWalkRet(EEventQueueWalkOk);
1296 // Check the WSEvent Type
1297 if (aQueueEvent->Type()==EEventPointer) // aEvent is a pointer event
1299 // It is a pointer event, so we can get the pointer number
1300 // to check if there is a repeat request for that pointer.
1301 const TInt eventPtrNum(TAdvancedPointerEventHelper::PointerNumber(*aQueueEvent));
1302 // If aEvent's pointer has an active repeat request,
1303 // then it'll have a repeat window.
1304 if ((eventPtrNum == aReqPtrNum) && RepeatWindow(eventPtrNum))
1306 // There is a repeat request for the pointer.
1307 // Is there a queued repeated event to be deleted?
1308 TWsPointer& wsPointer = iPointers[eventPtrNum];
1309 if (wsPointer.PointerEventRepeatCheck(aQueueEvent,aQueueEvent->Handle()))
1311 // Update the return value to purge the event
1312 // as it is a move/drag within the repeat rect
1313 eventQueueWalkRet=EEventQueueWalkDeleteEvent;
1317 // No queued repeated event was found, therefore the
1318 // request is still pending and needs to be cancelled.
1319 wsPointer.CancelPointerRepeatEventRequest();
1323 return eventQueueWalkRet;
1326 void TWsPointer::RequestRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
1328 CancelPointerRepeatEventRequest();
1329 iRepeatWindow=aWindow;
1330 iRepeatRect=aRequest.rect;
1331 iRepeatTimer->After(aRequest.time);
1332 aWindow->EventQueue()->WalkEventQueue(&PointerRepeatPurgeFunc,reinterpret_cast<TInt*>(iNumber));
1333 if (iRepeatWindow && !iRepeatRect.Contains(iPos-iRepeatWindow->Origin()))
1335 CancelPointerRepeatEventRequest();
1339 TInt TWsPointer::RequestPointerRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
1341 TInt errNo = KErrNone;
1342 TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
1343 if(PointerNumberInRange(pointerNum))
1345 iPointers[pointerNum].RequestRepeatEvent(aWindow,aRequest);
1354 void TWsPointer::CancelPointerRepeatEventRequest()
1360 iRepeatTimer->Cancel();
1364 TInt TWsPointer::CancelPointerRepeatEventRequest(const TWsWinCmdCancelPointerRepeatEventRequest& aRequest)
1366 TInt errNo = KErrNone;
1367 TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
1368 if(PointerNumberInRange(pointerNum))
1370 iPointers[pointerNum].CancelPointerRepeatEventRequest();
1379 void TWsPointer::RepeatTimerCompleted()
1382 event.SetType(EEventPointer);
1384 TPoint3D point3D(iPos-iRepeatWindow->Origin());
1385 point3D.iZ=iPressureProximity;
1386 TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event,
1387 TPointerEvent::EButtonRepeat,
1388 TWindowServerEvent::GetModifierState(),
1390 (iPos-iRepeatWindow->BaseParent()->Origin()),
1392 QueuePointerEvent(iRepeatWindow, event);
1396 #if defined(__WINS__)
1397 void TWsPointer::SetXyInputType(TXYInputType aXyInputType)
1399 if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse)
1401 // change from Mouse types to Pointer/None
1402 for (TInt ii = 0; ii < iMaxPointers; ii++)
1404 if (iPointers[ii].iState != EPointerStateDown)
1406 iPointers[ii].iCurrentWindow=NULL;
1409 UpdatePointerCursor();
1411 else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer)
1413 // change from Pointer/None types to Mouse types
1414 for (TInt ii = 0; ii < iMaxPointers; ii++)
1416 if (iPointers[ii].iState != EPointerStateDown)
1418 TPoint pos(iPointers[ii].iPos);
1420 iPointers[ii].ReLogCurrentWindow(pos,parPos,NULL);
1423 UpdatePointerCursor();
1425 iXyInputType=aXyInputType;
1430 Updates Primary Pointer before aRawEvent is processed. Only events related to Primary
1431 Pointer will be sent to Clients which require single pointer environment.
1433 This method implements single pointer environment emulation rules (see design
1434 documentation for more details).
1436 @param aRawEvent Incoming event used to update the Primary Pointer. It must be a pointer event,
1437 as defined by TWsPointer::IsPointerEventType(TRawEvent::TType).
1439 void TWsPointer::UpdatePrimaryPointer(const TRawEvent& aRawEvent)
1441 iPreviousPrimaryPointer = iPrimaryPointer;
1443 TRawEvent::TType type=aRawEvent.Type();
1444 TInt pointerNumber = aRawEvent.PointerNumber();
1446 #ifdef LOG_WSERV_EVENTS
1447 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer Current Primary pointer = %d",iPrimaryPointer);
1448 for(TInt i=0; i< iMaxPointers; i++)
1450 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer Pointer Number= %d State = %x XY(%d,%d)",iPointers[i].iNumber,iPointers[i].iState,iPointers[i].iPos.iX,iPointers[i].iPos.iY);
1454 // If primary pointer is out of range, then the first pointer that will
1455 // start being detected (come back in range) will become primary.
1456 if (iPointers[iPrimaryPointer].iState == EPointerStateOutOfRange)
1458 if (type != TRawEvent::EPointer3DOutOfRange && iPointers[pointerNumber].iState == EPointerStateOutOfRange)
1460 iPrimaryPointer = pointerNumber;
1461 #ifdef LOG_WSERV_EVENTS
1462 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(OutRange) = %d",iPrimaryPointer);
1468 // if non-primary pointer sends EButton1Down event, and actual primary pointer
1469 // is not down, then the pointer which has sent EButton1Down becomes primary.
1470 if (type == TRawEvent::EButton1Down &&
1471 iPointers[iPrimaryPointer].iState != EPointerStateDown)
1473 iPrimaryPointer = pointerNumber;
1474 #ifdef LOG_WSERV_EVENTS
1475 RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(ButtonDown) = %d",iPrimaryPointer);
1481 /** Sets Z coordinate threshold values for TPointerEvent::EEnterCloseProximity
1482 and TPointerEvent::EExitCloseProximity events.
1483 @return KErrNone if successful,
1484 KErrNotSupported if the device doesn't support threshold values,
1485 KErrArgument if aEnterCloseProximityThreshold is less than aExitCloseProximityThreshold
1486 @see RWsSession::SetCloseProximityThresholds which calls this method
1488 TInt TWsPointer::SetCloseProximityThresholds(TInt aEnterCloseProximityThreshold, TInt aExitCloseProximityThreshold)
1490 if (aEnterCloseProximityThreshold < aExitCloseProximityThreshold)
1492 return KErrArgument;
1495 TInt ret = HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, aEnterCloseProximityThreshold);
1496 if (ret != KErrNone)
1501 ret = HAL::Set(HALData::EPointer3DExitCloseProximityThreshold, aExitCloseProximityThreshold);
1502 WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicProxThresholdsInconsist);
1503 if (ret != KErrNone)
1505 HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, iEnterCloseProximityThreshold);
1509 iEnterCloseProximityThreshold = aEnterCloseProximityThreshold;
1510 iExitCloseProximityThreshold = aExitCloseProximityThreshold;
1515 @return Z coordinate threshold value for TPointerEvent::EEnterCloseProximity events
1516 @see RWsSession::GetEnterCloseProximityThreshold which calls this method
1518 TInt TWsPointer::GetEnterCloseProximityThreshold()
1520 return iEnterCloseProximityThreshold;
1524 @return Z coordinate threshold value for TPointerEvent::EExitCloseProximity events
1525 @see RWsSession::GetExitCloseProximityThreshold which calls this method
1527 TInt TWsPointer::GetExitCloseProximityThreshold()
1529 return iExitCloseProximityThreshold;
1532 /** Sets Z coordinate threshold value for TPointerEvent::EEnterHighPressure and
1533 TPointerEvent::EExitHighPressure events.
1534 @return KErrNone if successful,
1535 KErrNotSupported if the device doesn't support threshold values,
1536 KErrArgument if aEnterHighPressureThreshold is less than aExitHighPressureThreshold
1537 @see RWsSession::SetHighPressureThresholds which calls this method
1539 TInt TWsPointer::SetHighPressureThresholds(TInt aEnterHighPressureThreshold, TInt aExitHighPressureThreshold)
1541 if (aEnterHighPressureThreshold < aExitHighPressureThreshold)
1543 return KErrArgument;
1546 TInt ret = HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, aEnterHighPressureThreshold);
1547 if (ret != KErrNone)
1552 ret = HAL::Set(HALData::EPointer3DExitHighPressureThreshold, aExitHighPressureThreshold);
1553 WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicPressThresholdsInconsist);
1554 if (ret != KErrNone)
1556 HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, iEnterHighPressureThreshold);
1560 iEnterHighPressureThreshold = aEnterHighPressureThreshold;
1561 iExitHighPressureThreshold = aExitHighPressureThreshold;
1566 @return Z coordinate threshold value for TPointerEvent::EEnterHighPressure events
1567 @see RWsSession::GetEnterHighPressureThreshold which calls this method
1569 TInt TWsPointer::GetEnterHighPressureThreshold()
1571 return iEnterHighPressureThreshold;
1575 @return Z coordinate threshold value for TPointerEvent::EExitHighPressure events
1576 @see RWsSession::GetExitHighPressureThreshold which calls this method
1578 TInt TWsPointer::GetExitHighPressureThreshold()
1580 return iExitHighPressureThreshold;
1585 This function is specific for capacitive touch screens, where user's finger is the pointer device.
1586 Usability studies have shown that the user's perception of the location of the pointer hit is always
1587 away from few pixels north of the actual hit centre as detected by the digitizer device. So, this function
1588 will shift all pointer events by a specified Y displacement.
1590 @param aY Current y coordinate pointer position.
1593 void TWsPointer::ShiftYCoordinate(TInt& aY)
1595 WS_ASSERT_DEBUG(iYOffset>=0, EWsPanicInvalidPointerOffset);
1606 CWsPointerTimer::CWsPointerTimer(MPointerTimerCallback& aPointerTimerCallback)
1607 : CTimer(EPointerRepeatPriority), iPointerTimerCallback(aPointerTimerCallback)
1610 void CWsPointerTimer::ConstructL()
1612 CTimer::ConstructL();
1613 CActiveScheduler::Add(this);
1616 CWsPointerTimer* CWsPointerTimer::NewL(MPointerTimerCallback& aWsPointer)
1618 CWsPointerTimer* self = new(ELeave) CWsPointerTimer(aWsPointer);
1619 CleanupStack::PushL(self);
1621 CleanupStack::Pop(self);
1625 void CWsPointerTimer::RunL()
1627 User::ResetInactivityTime();
1628 WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
1629 iPointerTimerCallback.RepeatTimerCompleted();
1633 CWsPointerBuffer::~CWsPointerBuffer()
1635 if (this == iCurrentBuffer)
1637 // We're about to be destroyed - don't want to be pointed at any more.
1638 iCurrentBuffer = NULL;
1640 iList.Remove(*this);
1643 void CWsPointerBuffer::ConnectL(CWsClientWindow* aWindow, TInt aMaxPoints, TUint aFlags)
1645 CWsPointerBuffer* pb=NULL;
1646 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1648 if (pb->iWindow==aWindow)
1650 User::Leave(KErrInUse);
1653 CWsPointerBuffer* pbuf=new(ELeave) CWsPointerBuffer;
1654 pbuf->iWindow=aWindow;
1655 pbuf->iMaxPoints=aMaxPoints;
1656 pbuf->iFlags=aFlags;
1657 iList.AddFirst(*pbuf);
1658 CleanupStack::PushL(pbuf);
1660 CleanupStack::Pop();
1663 void CWsPointerBuffer::Disconnect(CWsClientWindow* aWindow)
1665 CWsPointerBuffer* pb=NULL;
1666 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1668 if (pb->iWindow==aWindow)
1670 delete pb; // Note that the destructor also sets iCurrentBuffer to NULL if it is pointing at pb
1671 TRAP_IGNORE(AdjustMaxSizeL()); // Shouldn't fail, but doesn't matter if it does as we simply have a larger buffer than needed
1672 break; // from for loop
1677 void CWsPointerBuffer::Reset()
1680 iPointerBuffer->Reset();
1683 void CWsPointerBuffer::SignalBufferReady()
1687 if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
1694 void CWsPointerBuffer::PointerEvent(CWsClientWindow* aWindow,const TPoint &aPoint)
1696 if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
1699 CWsPointerBuffer* pb=NULL;
1700 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1702 if (pb->iWindow==aWindow)
1705 break; // from for loop
1709 iPointerBuffer->Add(&aPoint);
1710 SignalBufferReady();
1713 void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow* aWindow,TInt aMaxPoints)
1715 enum {KPointerMoveBufferSize=32}; // Puts 256 bytes on the stack
1716 if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1719 TInt max=Min(aMaxPoints,iPointerBuffer->Count());
1721 aWindow->WsOwner()->SetReply(max);
1723 TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
1724 for(TInt index=0;index<max;index++)
1726 iPointerBuffer->Remove(&point);
1727 pnts.Append((TUint8 *)&point,sizeof(TPoint));
1729 if (buflen==KPointerMoveBufferSize)
1731 CWsClient::ReplyBuf(pnts);
1738 CWsClient::ReplyBuf(pnts);
1740 if (iPointerBuffer->Count())
1742 SignalBufferReady();
1747 void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
1749 if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
1755 void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow* aWindow)
1757 if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1763 void CWsPointerBuffer::AdjustMaxSizeL()
1766 CWsPointerBuffer* pb=NULL;
1767 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1769 if (pb->iMaxPoints>max)
1776 delete iPointerBuffer;
1777 iPointerBuffer=NULL;
1779 else if (!iPointerBuffer)
1781 CCirBuf<TPoint>* pointerBuffer=new(ELeave) CCirBuf<TPoint>;
1782 CleanupStack::PushL(pointerBuffer);
1783 pointerBuffer->SetLengthL(max);
1784 CleanupStack::Pop();
1785 iPointerBuffer=pointerBuffer;
1789 iPointerBuffer->SetLengthL(max);