os/graphics/windowing/windowserver/nga/SERVER/POINTER.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Pointer functions
    15 // 
    16 //
    17 
    18 #include <e32std.h>
    19 #include <e32hal.h>
    20 #include "W32CLICK.H"
    21 #include "pointer.h"
    22 #include "rootwin.h"
    23 #include "windowgroup.h"
    24 #include "KEYCLICK.H"
    25 #include "ScrDev.H"
    26 #include "EVENT.H"
    27 #include "panics.h"
    28 #include "wstop.h"
    29 #include "inifile.h"
    30 #include <hal.h>
    31 #include "advancedpointereventhelper.h"
    32 #include "graphics/pointereventdata.h"
    33 #include "debughelper.h"
    34 
    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;
    58 #if defined(__WINS__)
    59 TBool               TWsPointer::iEmulatorRotatePointerCoords;
    60 #endif
    61 
    62 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_SwEvent,ECapabilitySwEvent);
    63 
    64 void TWsPointer::InitStaticsL()
    65 	{
    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).
    69 	iYOffset = 0;
    70     _LIT( KWSERVIniFileVarYShifting, "YSHIFTING");
    71     TBool fetchingSucceeded = WsIniFile->FindVar(KWSERVIniFileVarYShifting, iYOffset);
    72     WS_ASSERT_ALWAYS(iYOffset>=0, EWsPanicInvalidPointerOffset);
    73     if ( !fetchingSucceeded )
    74        {
    75        iYOffset = 0;
    76        }
    77 
    78 #if defined(__WINS__)    
    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);
    83 #endif
    84 
    85 	const CScreen* screen = CWsTop::Screen();
    86 	WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
    87 	iRootWindow = screen->RootWindow();
    88 	
    89 	TMachineInfoV1Buf machineInfo;
    90 	UserHal::MachineInfo(machineInfo);
    91 	iXyInputType=machineInfo().iXYInputType;
    92 	
    93 	// Read EPointerMaxPointers from HAL and check if reported value is consistent
    94 	// with iXyInputType.
    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)
    99 		{
   100 		iMaxPointers = 1;
   101 		}
   102 	else
   103 		{
   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)
   109 			{
   110 			iMaxPointers = 1;
   111 			}
   112 		}
   113 	
   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);
   117 #endif
   118 	
   119 	// Does device support Z coordinate of the pointers?
   120 	if(HAL::Get(HALData::EPointer3D,iIs3DPointer)!=KErrNone)
   121 		{
   122 		iIs3DPointer=EFalse; // No API, then no 3D pointers
   123 		}
   124 	WS_ASSERT_ALWAYS(!iIs3DPointer || XyInput(), EWsPanicPointer3DInconsistent);
   125 	
   126 #ifdef LOG_WSERV_EVENTS
   127 	RDebug::Printf("_WSEVENT_POINTER: Z coordinate supported %d", iIs3DPointer);
   128 #endif	
   129 	
   130 	// Initialize thresholds for EEnterCloseProximity, EExitCloseProximity,
   131 	// EEnterHighPressure and EExitHightPressure events.
   132 	if(HAL::Get(HALData::EPointer3DEnterCloseProximityThreshold,
   133 			iEnterCloseProximityThreshold) != KErrNone)
   134 		{
   135 		iEnterCloseProximityThreshold = KMaxTInt;
   136 		}
   137 	if(HAL::Get(HALData::EPointer3DExitCloseProximityThreshold,
   138 			iExitCloseProximityThreshold) != KErrNone)
   139 		{
   140 		iExitCloseProximityThreshold  = KMinTInt;
   141 		}
   142 	if(HAL::Get(HALData::EPointer3DEnterHighPressureThreshold,
   143 			iEnterHighPressureThreshold)  != KErrNone)
   144 		{
   145 		iEnterHighPressureThreshold   = KMaxTInt;
   146 		}
   147 	if(HAL::Get(HALData::EPointer3DExitHighPressureThreshold,
   148 			iExitHighPressureThreshold)   != KErrNone)
   149 		{
   150 		iExitHighPressureThreshold    = KMinTInt;
   151 		}
   152 	
   153 	iPointers = RArray<TWsPointer>(iMaxPointers);
   154 	TWsPointer emptyPointer;	
   155 	emptyPointer.iRepeatTimer = NULL;
   156 	emptyPointer.Clear();
   157 	for (TInt ii = 0; ii < iMaxPointers; ii++)
   158 		{
   159 		emptyPointer.iNumber = ii;
   160 		User::LeaveIfError(iPointers.Append(emptyPointer));
   161 		RepeatTimer(ii) = CWsPointerTimer::NewL(iPointers[ii]);
   162 		}
   163 	
   164 	iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
   165 	}
   166 
   167 void TWsPointer::Clear()
   168 	{
   169 	iState = EPointerStateOutOfRange;
   170 	iPos.iX = 0;
   171 	iPos.iY = 0;
   172 	iPressureProximity = 0;
   173 	iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
   174 	iActualWinPointerIsOver = NULL;
   175 	iGrabWindow = NULL;
   176 	iLastUnmatchedDown1 = NULL;
   177 	iLastUnmatchedDown2 = NULL;
   178 	iLastUnmatchedDown3 = NULL;
   179 	iLastUnmatchedEnterHighPressure = NULL;
   180 	iPrevClickWindow = NULL;
   181 	iInCloseProximity = EFalse;
   182 	iInHighPressure = EFalse;
   183 	CancelPointerRepeatEventRequest();	
   184 	}
   185 
   186 /**
   187 Turns off pointer cursor, deletes permenently pointer cursor update timer and pointer event repeat timer.
   188 */
   189 void TWsPointer::Stop()
   190 	{
   191 	SetPointerCursorMode(EPointerCursorNone);
   192 	UpdatePointerCursor();
   193 	delete iPeriodicTimer;
   194 	}
   195 
   196 void TWsPointer::DeleteStatics()
   197 	{
   198 	for (TInt ii = 0; ii < iMaxPointers; ii++)
   199 		{
   200 		delete RepeatTimer(ii);	
   201 		}		
   202 	iPointers.Close();
   203 	}
   204 
   205 void TWsPointer::SetPointerCursorPos(TPoint aPos)
   206 	{
   207 	RestrictPos(aPos,EFalse);
   208 	iPointers[iPrimaryPointer].iPos=aPos; 
   209 	iPointers[iPrimaryPointer].ReLogCurrentWindow();
   210 	UpdatePointerCursor();
   211 	}
   212 
   213 void TWsPointer::SendEnterExitEvent(TEventCode aType)
   214 	{
   215 	if (iCurrentWindow
   216 		&& (iState != EPointerStateOutOfRange)
   217 		&& (iNumber == iPrimaryPointer || iCurrentWindow->AdvancedPointersEnabled())
   218 		&& !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
   219 		&& !iCurrentWindow->ShutDownInProgress())
   220 		{
   221 		iCurrentWindow->QueueEvent(aType, iNumber);
   222 		}
   223 	}
   224 
   225 void TWsPointer::SetCurrentWindow(const CWsWindow* aWin)
   226 	{
   227 	if (aWin!=iCurrentWindow)
   228 		{
   229 		SendEnterExitEvent(EEventPointerExit);
   230 		iCurrentWindow=aWin;
   231 		SendEnterExitEvent(EEventPointerEnter);
   232 		}
   233 	}
   234 
   235 /**
   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. 
   239 */
   240 void TWsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup* aForceInGroup)
   241 	{
   242 	if (iRootWindow)
   243 		{
   244 		SetCurrentWindow(iRootWindow->PointerWindow(iPos,&aPos,&aParentPos,iGrabWindow,
   245 				                                    iActualWinPointerIsOver,aForceInGroup));
   246 		}
   247 	else
   248 		{
   249 		iCurrentWindow=NULL;
   250 		}
   251 	}
   252 
   253 /**
   254 Relog all pointer's current windows when the window layout has changed.
   255 */
   256 void TWsPointer::ReLogPointersCurrentWindows()
   257 	{
   258 	for (TInt ii = 0; ii < iMaxPointers; ii++)
   259 		{
   260 		TWsPointer& pointer = iPointers[ii];
   261 		pointer.ReLogCurrentWindow();
   262 		}
   263 	}
   264 
   265 /*
   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.
   269 */
   270 void TWsPointer::ReLogCurrentWindow()
   271 	{
   272 	if (iCurrentWindow)
   273 		{
   274 		SetCurrentWindow(iRootWindow->PointerWindow(iPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
   275 		}
   276 	}
   277 
   278 /*
   279 Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
   280 is the current window
   281 */
   282 void TWsPointer::ReLogWindow(const CWsWindow* aWin)
   283 	{
   284 	for (TInt ii = 0; ii < iMaxPointers; ii++)
   285 		{
   286 		if (aWin == iPointers[ii].iCurrentWindow)
   287 			{
   288 			iPointers[ii].SendEnterExitEvent(EEventPointerEnter);
   289 			}
   290 		}
   291 	}
   292 
   293 void TWsPointer::UnmatchedEventPurged(TPointerEvent::TType aPointerType, TUint aHandle)
   294 	{
   295 	if (aPointerType==TPointerEvent::EButton1Up)
   296 		{
   297 		if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
   298 			{
   299 			iGrabWindow=NULL;
   300 			}
   301 		if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
   302 			{
   303 			CancelPointerRepeatEventRequest();
   304 			}
   305 		}
   306 	switch(aPointerType)
   307 		{
   308 		case TPointerEvent::EButton1Up:
   309 			iLastUnmatchedDown1=aHandle;
   310 			break;
   311 		case TPointerEvent::EButton2Up:
   312 			iLastUnmatchedDown2=aHandle;
   313 			break;
   314 		case TPointerEvent::EButton3Up:
   315 			iLastUnmatchedDown3=aHandle;
   316 			break;
   317 		case TPointerEvent::EExitHighPressure:
   318 			iLastUnmatchedEnterHighPressure=aHandle;
   319 			break;
   320 		default:;
   321 		}
   322 	}
   323 
   324 void TWsPointer::WindowDisconnected(const CWsWindow* deletedWindow)
   325 	{
   326 	for (TInt pointerNum = 0; pointerNum < iMaxPointers; pointerNum++)
   327 		{
   328 		TWsPointer& pointer = iPointers[pointerNum];
   329 		if (pointer.iRepeatWindow==deletedWindow)
   330 			{
   331 			pointer.CancelPointerRepeatEventRequest();
   332 			}		
   333 		if (pointer.iGrabWindow==deletedWindow)
   334 			{
   335 			pointer.iGrabWindow=NULL;
   336 			}
   337 		if (pointer.iCurrentWindow==deletedWindow)
   338 			{
   339 			pointer.ReLogCurrentWindow();
   340 			if (pointerNum == iPrimaryPointer)
   341 				{
   342 				UpdatePointerCursor();
   343 				}
   344 			}
   345 		}
   346 	}
   347 
   348 /*
   349 Callback function pointer for up event remove event queue walk
   350 */
   351 TEventQueueWalkRet RemovePointerUpFunc(TAny* aHandle, TWsEvent* aEvent)
   352 	{
   353 	if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
   354 		{
   355 		return(EEventQueueWalkDeleteEvent);
   356 		}
   357 	return(EEventQueueWalkOk);
   358 	}
   359 
   360 /*
   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.
   365 
   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.
   370 */
   371 TInt TWsPointer::ClaimGrab(const CWsWindow *aWindow,const TWsWinCmdGrabControl& aGrabControl)
   372 	{
   373 	TInt pointerNumber(aGrabControl.HasPointerNumber() ? aGrabControl.pointerNumber : iPrimaryPointer);
   374 	TBool advancedPointersEnabled(aWindow->AdvancedPointersEnabled());
   375 	TInt errNo(KErrNone);
   376 
   377 	if(!advancedPointersEnabled	&& aGrabControl.HasPointerNumber() && (TAdvancedPointerEvent::EDefaultPointerNumber!=pointerNumber))
   378 		{
   379 		// The window is in emulation mode, and cannot get events for anything other than the primary pointer		
   380 		errNo=KErrNotSupported;
   381 		}
   382 	else if(advancedPointersEnabled && ((pointerNumber<0) || (pointerNumber>=iMaxPointers)))
   383 		{
   384 		// Unknown pointer number
   385 		errNo=KErrNotFound;
   386 		}
   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")))
   389 		{
   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;
   394 		}
   395 	else
   396 		{
   397 		iPointers[pointerNumber].ClaimGrab(aWindow,aGrabControl.CheckFlags(TWsWinCmdGrabControl::ESendUpEvent));		
   398 		}	
   399 	
   400 	return errNo;
   401 	}
   402 
   403 /*
   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.
   407 */
   408 void TWsPointer::ClaimGrab(const CWsWindow* aWindow,TBool aSendUpEvent)
   409 	{	
   410 	TInt modState=TWindowServerEvent::GetModifierState();
   411 	TWsEvent event;
   412 	TAdvancedPointerEvent& pointerEvent=*event.Pointer();
   413 	TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, TPointerEvent::EButton1Up,modState,TPoint3D(iPos.iX,iPos.iY,iPressureProximity),iNumber);
   414 	if (iState == EPointerStateDown)
   415 		{
   416 		if (iCurrentWindow!=aWindow)
   417 			{
   418 			if (aSendUpEvent)
   419 				{
   420 				ProcessEvent(event,NULL,EFalse);
   421 				}
   422 			else // If up event already in queue purge it
   423 				{
   424 				CEventQueue* eventQueue = iCurrentWindow->EventQueue();
   425 				if (eventQueue)
   426 					{
   427 					TUint handle=iCurrentWindow->ClientHandle();
   428 					eventQueue->WalkEventQueue(&RemovePointerUpFunc,&handle);
   429 					}
   430 				}
   431 			iState = EPointerStateDown;
   432 			if (aWindow->HasPointerGrab())
   433 				{
   434 				iGrabWindow=aWindow;
   435 				}
   436 			ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
   437 			pointerEvent.iType=TPointerEvent::EDrag;
   438 			ProcessPointerEvent(event);
   439 			}
   440 		}
   441 	else if (iState == EPointerStateUp)
   442 		{
   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);
   449 		iGrabWindow=NULL;
   450 		iCurrentWindow=current;
   451 		}
   452 	}
   453 
   454 /**
   455 @return ETrue if matching event has been purged, so current event should not be delivered,
   456         EFalse otherwise.
   457 */
   458 TBool TWsPointer::CheckMatchingEventPurged(TPointerEvent::TType aType)
   459 	{
   460 	switch(aType)
   461 		{
   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())
   481 			{
   482 			return ETrue; // Don't deliver the event as we've already thrown away matching event
   483 			}
   484 		default:		//Should never get to default
   485 			break;
   486 		}
   487 	return EFalse;
   488 	}
   489 
   490 TBool TWsPointer::QueuePointerEvent(const CWsWindow* aWindow, TWsEvent &aEvent)
   491 	{
   492 	if (aWindow->WsOwner() &&
   493 		(aWindow->AdvancedPointersEnabled() || 
   494 		TAdvancedPointerEventHelper::PointerNumber(aEvent) == iPrimaryPointer))
   495 		{
   496 		CEventQueue* queue=aWindow->EventQueue();
   497 		aEvent.SetHandle(aWindow->ClientHandle());
   498 #ifdef LOG_WSERV_EVENTS
   499 		RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent with AdvancedPointerEnabled");
   500 #endif
   501 		if (aEvent.Handle()!=0)
   502 			{
   503  			if(!aWindow->AdvancedPointersEnabled())
   504  				{
   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
   507  				// being different.
   508  				TAdvancedPointerEventHelper::SetPointerNumber(aEvent,TAdvancedPointerEvent::EDefaultPointerNumber);
   509  				aEvent.Pointer()->iModifiers&=~EModifierAdvancedPointerEvent;  // Clear the advanced pointer flag
   510  				}
   511 			if (queue->UpdateLastPointerEvent(aEvent))
   512 				{
   513 				return EFalse;
   514 				}
   515 			TWservEventPriorities priority=EEventPriorityLow;
   516 			switch (aEvent.Pointer()->iType)
   517 				{
   518 				case TPointerEvent::EButton1Up:
   519 				case TPointerEvent::EButton2Up:
   520 				case TPointerEvent::EButton3Up:
   521 				case TPointerEvent::EExitHighPressure:
   522 					if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
   523 						{
   524 						#ifdef LOG_WSERV_EVENTS
   525 						RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 01");
   526 						#endif
   527 						return ETrue;
   528 						}
   529 					if (queue->CheckRoom())
   530 						{
   531 						if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
   532 							{
   533 							#ifdef LOG_WSERV_EVENTS
   534 							RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 02");
   535 							#endif
   536 							return ETrue;
   537 							}
   538 						}
   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;
   546 					break;
   547 				default:;
   548 				}
   549 #ifdef LOG_WSERV_EVENTS
   550 			RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent After adding event to clientqueue Event State %d ", iState);
   551 #endif
   552 			queue->QueueEvent(aEvent,priority);
   553 			}
   554 		}
   555 	return EFalse;
   556 	}
   557 
   558 /*
   559 Moves the window group which contains this pointer's iCurrentWindow on top.
   560 */
   561 void TWsPointer::ProcessForegroundCheck()
   562 	{
   563 	CWsWindowGroup* group=((CWsTopClientWindow *)iCurrentWindow)->TopClientWindow()->Parent();
   564 	if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
   565 		{
   566 		group->SetOrdinalPosition(0);
   567 		}
   568 	}
   569 
   570 /*
   571 Pointer Event Processing - stage 3 of 3:
   572 - setting event's time
   573 - auto foreground check
   574 - double clicks detection
   575 - pointer repeats
   576 - add event to client's queue
   577 - drag&drop capturing
   578 */
   579 void TWsPointer::ProcessPointerEvent(TWsEvent& aEvent)
   580 	{
   581 	if (iCurrentWindow && iCurrentWindow!=iRootWindow)
   582 		{
   583 		aEvent.SetType(EEventPointer);
   584 		aEvent.SetTimeNow();
   585 		TPointerEvent::TType type=aEvent.Pointer()->iType;
   586 		switch(type)
   587 			{
   588 			//TUint lastUnmatchedDown;
   589 			case TPointerEvent::EButton1Down:
   590 				ProcessForegroundCheck(); 
   591 				/*Fall Through*/
   592 			case TPointerEvent::EButton2Down:
   593 			case TPointerEvent::EButton3Down:
   594 				{
   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))
   600 					{
   601 					aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
   602 					iPrevClickWindow=NULL;	// Set to NULL to block a double double click
   603 					}
   604 				else
   605 					{
   606 					iPrevClickWindow=iCurrentWindow;
   607 					}
   608 				iPrevClickEventType=type;
   609 				iPrevClickPos=pos;
   610 				iPrevClickTime=aEvent.Time();
   611 				}
   612 				break;
   613 			default:
   614 				break;
   615 			}
   616 		if (iRepeatWindow)
   617 			{
   618 			if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
   619 				{
   620 				return;
   621 				}
   622 			CancelPointerRepeatEventRequest();
   623 			}
   624 		if (QueuePointerEvent(iCurrentWindow, aEvent))
   625 			{
   626 			return;
   627 			}
   628 		if (iCurrentWindow->DragDropCapture())
   629 			{
   630 			aEvent.SetType(EEventDragDrop);
   631 			QueuePointerEvent(iActualWinPointerIsOver, aEvent);
   632 			}
   633 		}
   634 	}
   635 
   636 TInt PointerTimerCallBack(TAny *)
   637 	{
   638 	TWsPointer::TimerExpired();
   639 	return(KErrNone);
   640 	}
   641 
   642 void TWsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
   643 	{
   644 	CScreen* screen = iRootWindow->Screen();
   645 	WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
   646 #if defined(__WINS__)
   647 	if (aWithinDrawableArea)
   648 		{
   649 	if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
   650 			{
   651 		return;			//Not in the drawable area so user may be trying to click on facia button.
   652 			}
   653 		}
   654 #endif
   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;
   664 	}
   665 
   666 #if defined(__WINS__)
   667 TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
   668 #else
   669 TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent)
   670 #endif
   671 	{
   672 	TRawEvent::TType type=aRawEvent.Type();
   673 	
   674 	if (!IsPointerEventType(type)) 
   675 		return ETrue;
   676 	
   677 	if (!XyInput())
   678 		{
   679 		return EFalse;
   680 		}
   681 	
   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);
   686 #endif
   687 	
   688 	// check correctness of aRawEvent.PointerNumber()
   689 	if (iMaxPointers > 1)
   690 		{
   691 		if (aRawEvent.PointerNumber() >= iMaxPointers)
   692 			{
   693 			return EFalse;
   694 			}
   695 		}
   696 	else
   697 		{
   698 		aRawEvent.SetPointerNumber(0);
   699 		}
   700 	
   701 	if (type != TRawEvent::EPointer3DOutOfRange)
   702 		{
   703 		// operations on coordinates are valid for all types except EPointer3DOutOfRange
   704 		TPoint xy=aRawEvent.Pos();
   705 		if (DeltaMouse())
   706 			{
   707 	#if defined(__WINS__)
   708 			if (aFromHardware)
   709 				return EFalse;
   710 	#endif
   711 			if (type==TRawEvent::EPointerMove)
   712 				{
   713 				xy+=iPointers[aRawEvent.PointerNumber()].iPos;
   714 				ShiftYCoordinate(xy.iY);
   715 				RestrictPos(xy);
   716 				}
   717 			else
   718 				xy=iPointers[aRawEvent.PointerNumber()].iPos;
   719 			}
   720 		else
   721 			{
   722 	#if !defined(__WINS__)
   723 			TranslateCoordsOnRotation(xy);
   724 	#else
   725 			if(iEmulatorRotatePointerCoords)
   726 				{
   727 				//emulators that support rotated drawing and touch may want to enable 
   728 				//rotation of pointer events in wsini.
   729 				TranslateCoordsOnRotation(xy);
   730 				}
   731 	#endif
   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);
   736 			RestrictPos(xy);
   737 			}
   738 		aRawEvent.Set(type, xy.iX, xy.iY, 
   739 					  iIs3DPointer ? aRawEvent.Pos3D().iZ : 0);
   740 		}
   741 	
   742 #ifdef LOG_WSERV_EVENTS
   743 	RDebug::Printf("_WSEVENT_POINTER: Coordinates after Rotation and shift [%d, %d]", 
   744 							aRawEvent.Pos().iX, aRawEvent.Pos().iY);
   745 #endif
   746 	return ETrue;
   747 	}
   748 
   749 void TWsPointer::TranslateCoordsOnRotation(TPoint& aPoint)
   750 	{
   751 	CScreen* screen=iRootWindow->Screen();
   752 	TSize screenSize=screen->SizeInPixels()-TSize(1,1);     //This is in the current rotation
   753 	switch (screen->Orientation())
   754 		{
   755 		case CFbsBitGc::EGraphicsOrientationRotated90:
   756 			aPoint.SetXY(aPoint.iY,screenSize.iHeight-aPoint.iX);
   757 			break;
   758 		case CFbsBitGc::EGraphicsOrientationRotated180:
   759 			aPoint=-(aPoint-screenSize);
   760 			break;
   761 		case CFbsBitGc::EGraphicsOrientationRotated270:
   762 			aPoint.SetXY(screenSize.iWidth-aPoint.iY,aPoint.iX);
   763 			break;
   764 		default:;       //To stop warning
   765 		}
   766 	}
   767 
   768 /**
   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.
   776  */
   777 TBool TWsPointer::PreProcessClientEvent(TRawEvent &aRawEvent, TBool aAdvancedPointersEnabled)
   778 	{
   779 	TRawEvent::TType type=aRawEvent.Type();
   780 	
   781 	if (!IsPointerEventType(type)) 
   782 		return ETrue;
   783 	
   784 	// validate pointer number
   785 	if (aAdvancedPointersEnabled)
   786 		{
   787 		// ignore event if both digitizer driver and Client support pointer numbers, but number is invalid
   788 		if (aRawEvent.PointerNumber() >= iMaxPointers)
   789 			{
   790 			return EFalse;
   791 			}		
   792 		}
   793 	else
   794 		{
   795 		// set to iPrimaryPointer if Client doesn't support pointer numbers
   796 		aRawEvent.SetPointerNumber(iPrimaryPointer);
   797 		}
   798 	
   799 	// validate Z coordinate for all pointer events except EPointer3DOutOfRange
   800 	if (type != TRawEvent::EPointer3DOutOfRange)
   801 		{
   802 		if (!iIs3DPointer)
   803 			{
   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);
   807 			}
   808 		else if (!aAdvancedPointersEnabled)
   809 			{
   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);
   813 			}
   814 		}
   815 	
   816 	return ETrue;
   817 	}
   818 
   819 TBool TWsPointer::IsPointerEventType(TRawEvent::TType aType)
   820 	{
   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);
   827 #endif
   828 	return (aType == TRawEvent::EPointerMove) ||
   829 		   (aType == TRawEvent::EPointerSwitchOn) ||
   830 		   (aType >= TRawEvent::EButton1Down && aType <= TRawEvent::EButton3Up) ||
   831 		   (aType == TRawEvent::EPointer3DOutOfRange);
   832 	}
   833 
   834 /*
   835 Pointer Event processing.
   836 
   837 This method redirects pointer event processing to proper TWsPointer object.
   838 */
   839 void TWsPointer::ProcessWsEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
   840 	{
   841 	if(iPrimaryPointer!=iPreviousPrimaryPointer)
   842 		{
   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);
   847 		}
   848 	
   849 	if (aEvent.Pointer()->iType == TPointerEvent::EOutOfRange)
   850 		{
   851 		iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessOutOfRangeEvent(aEvent, aForceInGroup, aNatural);
   852 		}
   853 	else
   854 		{
   855 		iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessEvent(aEvent, aForceInGroup, aNatural);
   856 		}
   857 	}
   858 
   859 /*
   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
   864 - simulated moves
   865 - generating events: EEnterCloseProximity, EExitCloseProximity, EEnterHighPressure, EExitHighPressure
   866 */
   867 void TWsPointer::ProcessEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
   868  	{
   869 	TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();
   870 	TPointerEvent::TType eventType=pointerEvent.iType;
   871 	
   872  	if (iState == EPointerStateOutOfRange)
   873  		{
   874  		// new pointer comes in range, so clear information after previous one 		
   875  		Clear();
   876  		}
   877  	
   878 	// update coordinates
   879 	iPos=pointerEvent.iPosition;
   880 	iPressureProximity=TAdvancedPointerEventHelper::Z(aEvent);
   881 	
   882 	if (eventType == TPointerEvent::EMove && !MovesAvailable() && iState != EPointerStateDown)
   883 		{
   884 		return;
   885 		}
   886 	
   887 	// re-log and update parent position
   888 	TPoint parPos;
   889 	ReLogCurrentWindow(pointerEvent.iPosition,parPos,aForceInGroup);
   890 	pointerEvent.iParentPosition=parPos;
   891 	
   892 	
   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);
   896 #endif
   897 	// update state
   898 	switch(eventType)
   899 		{
   900 		case TPointerEvent::EButton1Down:
   901 			if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
   902 				{
   903 				iGrabWindow=iCurrentWindow;
   904 				}
   905 			if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
   906 				{
   907 				pointerEvent.iType=TPointerEvent::EMove;
   908 				ProcessEvent(aEvent, EFalse);
   909 				pointerEvent.iType=TPointerEvent::EButton1Down;
   910 				}
   911 			switch(iState)
   912 				{
   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)
   921 						{
   922 						iInHighPressure = ETrue;
   923 						pointerEvent.iType = TPointerEvent::EEnterHighPressure;
   924 						ProcessEvent(aEvent, EFalse);
   925 						}
   926 					else
   927 						{
   928 						iInHighPressure = EFalse;
   929 						}
   930 					break;
   931 				case EPointerStateDown:
   932 					if (iInHighPressure && iPressureProximity < iExitHighPressureThreshold) 
   933 						{
   934 						iInHighPressure = EFalse;
   935 						eventType = TPointerEvent::EExitHighPressure;
   936 						}
   937 					else if (!iInHighPressure && iPressureProximity >= iEnterHighPressureThreshold)
   938 						{
   939 						iInHighPressure = ETrue;
   940 						eventType = TPointerEvent::EEnterHighPressure;
   941 						}
   942 					ProcessEvent(aEvent, aNatural);
   943 					break;
   944 				}
   945 			break;
   946 		case TPointerEvent::EButton1Up:
   947 			iGrabWindow=NULL;
   948 			switch(iState)
   949 				{
   950 				case EPointerStateDown:
   951 					iState = EPointerStateUp;
   952 					ProcessEvent(aEvent, aNatural);
   953 					if (iPressureProximity < iExitCloseProximityThreshold)
   954 						{
   955 						pointerEvent.iType = TPointerEvent::EExitCloseProximity;
   956 						iInCloseProximity = EFalse;
   957 						ProcessEvent(aEvent, EFalse);
   958 						}
   959 					else
   960 						{
   961 						iInCloseProximity = ETrue;
   962 						}
   963 					break;
   964 				case EPointerStateOutOfRange:
   965 					iState = EPointerStateUp;
   966 					SendEnterExitEvent(EEventPointerEnter);
   967 					// intentional lack of break statement
   968 				case EPointerStateUp:
   969 					if (iInCloseProximity &&
   970 						iPressureProximity < iExitCloseProximityThreshold)
   971 						{
   972 						iInCloseProximity = EFalse;
   973 						pointerEvent.iType = TPointerEvent::EExitCloseProximity;
   974 						}
   975 					else if (!iInCloseProximity &&
   976 							 iPressureProximity >= iEnterCloseProximityThreshold)
   977 						{
   978 						iInCloseProximity = ETrue;
   979 						pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
   980 						}
   981 					ProcessEvent(aEvent, aNatural);
   982 					break;
   983 				}
   984 			break;
   985 		case TPointerEvent::EMove:
   986 			switch(iState)
   987 				{
   988 				case EPointerStateDown:
   989 					if (iInHighPressure && 
   990 						iPressureProximity < iExitHighPressureThreshold) 
   991 						{
   992 						iInHighPressure = EFalse;
   993 						pointerEvent.iType = TPointerEvent::EExitHighPressure;
   994 						}
   995 					else if (!iInHighPressure &&
   996 							iPressureProximity >= iEnterHighPressureThreshold)
   997 						{
   998 						iInHighPressure = ETrue;
   999 						pointerEvent.iType = TPointerEvent::EEnterHighPressure;
  1000 						}
  1001 					else
  1002 						{
  1003 						pointerEvent.iType = TPointerEvent::EDrag;
  1004 						}
  1005 					break;
  1006 				case EPointerStateUp:
  1007 					if (iInCloseProximity &&
  1008 						iPressureProximity < iExitCloseProximityThreshold)
  1009 						{
  1010 						iInCloseProximity = EFalse;
  1011 						pointerEvent.iType = TPointerEvent::EExitCloseProximity;
  1012 						}
  1013 					else if (!iInCloseProximity &&
  1014 							 iPressureProximity >= iEnterCloseProximityThreshold)
  1015 						{
  1016 						iInCloseProximity = ETrue;
  1017 						pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
  1018 						}
  1019 					break;
  1020 				case EPointerStateOutOfRange:
  1021 					iState = EPointerStateUp;
  1022 					SendEnterExitEvent(EEventPointerEnter);
  1023 					if (iPressureProximity >= iEnterCloseProximityThreshold)
  1024 						{
  1025 						iInCloseProximity = ETrue;
  1026 						pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
  1027 						}
  1028 					break;
  1029 				}
  1030 			ProcessEvent(aEvent, aNatural);
  1031 			break;
  1032 		default:
  1033 			ProcessEvent(aEvent, aNatural);
  1034 			break;
  1035 		}
  1036 	}
  1037 
  1038 /**
  1039 Processes OutOfRange events:
  1040 - injects event to key click plugin
  1041 - directs event to the last current window
  1042 - sends Exit event
  1043 */
  1044 void TWsPointer::ProcessOutOfRangeEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup, TBool aNatural)
  1045 	{
  1046 	if (iState != EPointerStateOutOfRange)
  1047 		{
  1048 #ifdef LOG_WSERV_EVENTS
  1049        RDebug::Printf("_WSEVENT_POINTER: TWsPointer::ProcessOutOfRangeEvent Pointer Number = %d, iState =%d ", iNumber, iState); 
  1050 #endif
  1051 
  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);
  1055 		
  1056 		iState = EPointerStateOutOfRange;
  1057 		
  1058 		TAdvancedPointerEventHelper::SetPointerNumber(aEvent, iNumber);
  1059 		TAdvancedPointerEventHelper::SetZ(aEvent, iPressureProximity);
  1060 		
  1061 		TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();		
  1062 		iCurrentWindow=iRootWindow->PointerWindow(iPos,&pointerEvent.iPosition,&pointerEvent.iParentPosition,iGrabWindow,
  1063 			                   					  iActualWinPointerIsOver,aForceInGroup);
  1064 		ProcessEvent(aEvent, aNatural);
  1065 		}
  1066 	}
  1067 
  1068 void TWsPointer::NotifyCClick(TAdvancedPointerEvent& aPointerEvent)
  1069 	{
  1070 	if (CClick::IsHandler())
  1071 		{
  1072 		CClick::PointerEvent(iPos,aPointerEvent);
  1073 		TPointerEventData params;
  1074 		params.iVersion=0;
  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,&params);
  1083 		}
  1084 	}
  1085 
  1086 /*
  1087 Pointer Event Processing - stage 2 of 3:
  1088 - injecting events to key click plugin
  1089 - pointer filtering
  1090 - injection to event buffer
  1091 - injection to keyboard emulator
  1092 - clearing iCurrentWindow if going up and !MovesAvailable()
  1093 - updating pointer cursor
  1094 */
  1095 void TWsPointer::ProcessEvent(TWsEvent& aEvent, TBool aNatural)
  1096 	{
  1097 	if (aNatural)
  1098 		{
  1099 		NotifyCClick(*aEvent.Pointer());
  1100 		}
  1101 	
  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)))
  1106 		{
  1107 		if (iNumber == iPrimaryPointer)
  1108 			{
  1109 			TPoint pos=aEvent.Pointer()->iPosition;
  1110 			if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
  1111 				{
  1112 				CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
  1113 				}
  1114 			else if ((type==TPointerEvent::EEnterCloseProximity || type==TPointerEvent::EExitCloseProximity || 
  1115 					  type==TPointerEvent::EEnterHighPressure   || type==TPointerEvent::EExitHighPressure) && 
  1116 					 iCurrentWindow->UsingPointerBuffer())
  1117 				{
  1118 				CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
  1119 				ProcessPointerEvent(aEvent);
  1120 				}
  1121 			else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
  1122 				{
  1123 #ifdef LOG_WSERV_EVENTS
  1124 				RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for primary pointer");
  1125 #endif
  1126 				ProcessPointerEvent(aEvent);
  1127 				}
  1128 			}
  1129 		else if (!iCurrentWindow->UsingPointerBuffer() || (type != TPointerEvent::EMove && type != TPointerEvent::EDrag))
  1130 			{
  1131 #ifdef LOG_WSERV_EVENTS
  1132 			RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for non primary pointer");
  1133 #endif
  1134 			ProcessPointerEvent(aEvent);
  1135 			}
  1136 		}
  1137 	if (!MovesAvailable() && (type==TPointerEvent::EButton1Up || 
  1138 			                  type==TPointerEvent::ESwitchOn  ||
  1139 			                  type==TPointerEvent::EOutOfRange))
  1140 		{
  1141 		iCurrentWindow=NULL;
  1142 		}
  1143 	if (iNumber == iPrimaryPointer)
  1144 		{
  1145 		PointerCursorUpdateCheck();
  1146 		}
  1147 	}
  1148 
  1149 void TWsPointer::TimerExpired()
  1150 	{
  1151 	WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
  1152 	if (iUpdateRequired)
  1153 		{
  1154 		UpdatePointerCursor();
  1155 		iUpdateRequired=EFalse;
  1156 		}
  1157 	else
  1158 		{
  1159 		iTimerQueued=EFalse;
  1160 		iPeriodicTimer->Cancel();
  1161 		}
  1162 	}
  1163 
  1164 void TWsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
  1165 	{
  1166 	aTime=iDoubleClickMaxInterval;
  1167 	aDistance=iDoubleClickMaxDistance;
  1168 	}
  1169 
  1170 void TWsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
  1171 	{
  1172 	iDoubleClickMaxInterval=aTime;
  1173 	iDoubleClickMaxDistance=aDistance;
  1174 	}
  1175 
  1176 void TWsPointer::PointerCursorUpdateCheck()
  1177 	{
  1178 	CWsPointerCursor* sprite=CalculatePointerCursor();
  1179 	if (iCursorSprite || sprite)	// If there either was, or is a pointer cursor we need an update
  1180 		{
  1181 		if (!iTimerQueued)
  1182 			{
  1183 			UpdatePointerCursorTo(sprite);
  1184 			iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
  1185 									TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
  1186 									  TCallBack(PointerTimerCallBack,NULL));
  1187 			iTimerQueued=ETrue;
  1188 			}
  1189 		else
  1190 			{
  1191 			iUpdateRequired=ETrue;
  1192 			}
  1193 		}
  1194 	}
  1195 
  1196 void TWsPointer::UpdatePointerCursor()
  1197 	{
  1198 //__PROFILE_START(3);
  1199 	CWsPointerCursor* sprite=iPointers[iPrimaryPointer].CalculatePointerCursor();
  1200 	UpdatePointerCursorTo(sprite);
  1201 //__PROFILE_END(3);
  1202 	}
  1203 
  1204 void TWsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
  1205 	{
  1206 	if (iCursorSprite!=aNewCursor)
  1207 		{
  1208 		if (iCursorSprite)
  1209 			{
  1210 			iCursorSprite->Deactivate();
  1211 			}
  1212 		iCursorSprite=aNewCursor;
  1213 		if (iCursorSprite)
  1214 			{
  1215 			iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
  1216 			iCursorSprite->Activate();
  1217 			}
  1218 		}
  1219 	else if (iCursorSprite)
  1220 		{
  1221 		iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
  1222 		}
  1223 	}
  1224 
  1225 CWsPointerCursor* TWsPointer::CalculatePointerCursor()
  1226 	{
  1227 	CWsPointerCursor* sprite=NULL;
  1228 	if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
  1229 		{
  1230 		const CWsWindowBase* window=iCurrentWindow;
  1231 		do	{
  1232 			sprite=window->PointerCursor();
  1233 			if (window->WinType()!=EWinTypeClient)
  1234 				{
  1235 				break;
  1236 				}
  1237 			window=window->BaseParent();
  1238 			} while (!sprite);
  1239 		}
  1240 	if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
  1241 		{
  1242 		sprite=CWsClient::DefaultSystemPointerCursor();
  1243 		}
  1244 	return sprite;
  1245 	}
  1246 
  1247 /*
  1248 Callback function for event queue walk
  1249 */
  1250 TEventQueueWalkRet PointerRepeatPurgeFunc(TAny* aReqPtrNum, TWsEvent* aQueueEvent)
  1251 	{
  1252 	return(TWsPointer::PointerRepeatPurgeCheck(aQueueEvent, reinterpret_cast<TUint>(aReqPtrNum)));
  1253 	}
  1254 
  1255 TBool TWsPointer::PointerEventRepeatCheck(const TWsEvent* aEvent, TUint32 aHandle)
  1256 //
  1257 // Return ETrue if this pointer event is consumed by the pointer repeat
  1258 //
  1259 	{
  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);
  1263 	
  1264 	const TAdvancedPointerEvent* pntEvent=aEvent->Pointer();
  1265 	if (	TAdvancedPointerEventHelper::PointerNumber(*aEvent)==iNumber &&
  1266 			aHandle==iRepeatWindow->ClientHandle())
  1267 		{
  1268 		switch(pntEvent->iType)
  1269 			{
  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:
  1276 				{
  1277 				if(iRepeatRect.Contains(pntEvent->iPosition))
  1278 					{
  1279 					return(ETrue);
  1280 					}
  1281 				break;
  1282 				}
  1283 			default:
  1284 				// do nothing and drop through
  1285 				break;
  1286 			}		
  1287 		}
  1288 	return(EFalse);
  1289 	}
  1290 
  1291 TEventQueueWalkRet TWsPointer::PointerRepeatPurgeCheck(TWsEvent* aQueueEvent, TUint8 aReqPtrNum)
  1292 	{
  1293 	// Return value is "WalkOK", unless a repeated event is found that needs to be deleted.
  1294 	TEventQueueWalkRet eventQueueWalkRet(EEventQueueWalkOk);	
  1295 	
  1296 	// Check the WSEvent Type	
  1297 	if (aQueueEvent->Type()==EEventPointer) // aEvent is a pointer event
  1298 		{		
  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))
  1305 			{
  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()))
  1310 				{
  1311 				// Update the return value to purge the event
  1312 				// as it is a move/drag within the repeat rect
  1313 				eventQueueWalkRet=EEventQueueWalkDeleteEvent;
  1314 				}		
  1315 			else
  1316 				{
  1317 				// No queued repeated event was found, therefore the
  1318 				// request is still pending and needs to be cancelled.
  1319 				wsPointer.CancelPointerRepeatEventRequest();
  1320 				}
  1321 			}
  1322 		}
  1323 	return eventQueueWalkRet;
  1324 	}
  1325 
  1326 void TWsPointer::RequestRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
  1327 	{
  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()))
  1334 		{
  1335 		CancelPointerRepeatEventRequest();
  1336 		}	
  1337 	}
  1338 
  1339 TInt TWsPointer::RequestPointerRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
  1340 	{
  1341 	TInt errNo = KErrNone;
  1342 	TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
  1343 	if(PointerNumberInRange(pointerNum))
  1344 		{
  1345 		iPointers[pointerNum].RequestRepeatEvent(aWindow,aRequest);
  1346 		}
  1347 	else
  1348 		{
  1349 		errNo=KErrArgument;
  1350 		}
  1351 	return errNo;
  1352 	}
  1353 
  1354 void TWsPointer::CancelPointerRepeatEventRequest()
  1355 	{
  1356 	if (iRepeatWindow)
  1357 		{
  1358 		iRepeatWindow=NULL;
  1359 		if(iRepeatTimer)
  1360 			iRepeatTimer->Cancel();
  1361 		}
  1362 	}
  1363 
  1364 TInt TWsPointer::CancelPointerRepeatEventRequest(const TWsWinCmdCancelPointerRepeatEventRequest& aRequest)
  1365 	{
  1366 	TInt errNo = KErrNone;	
  1367 	TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
  1368 	if(PointerNumberInRange(pointerNum))
  1369 		{
  1370 		iPointers[pointerNum].CancelPointerRepeatEventRequest();
  1371 		}
  1372 	else
  1373 		{
  1374 		errNo=KErrArgument;
  1375 		}
  1376 	return errNo;
  1377 	}
  1378 
  1379 void TWsPointer::RepeatTimerCompleted()
  1380 	{
  1381 	TWsEvent event;
  1382 	event.SetType(EEventPointer);
  1383 	event.SetTimeNow();
  1384 	TPoint3D point3D(iPos-iRepeatWindow->Origin());
  1385 	point3D.iZ=iPressureProximity;
  1386 	TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event,
  1387 														  TPointerEvent::EButtonRepeat,
  1388 														  TWindowServerEvent::GetModifierState(),
  1389 														  point3D,
  1390 														  (iPos-iRepeatWindow->BaseParent()->Origin()),
  1391 														  iNumber);
  1392 	QueuePointerEvent(iRepeatWindow, event);
  1393 	iRepeatWindow=NULL;
  1394 	}
  1395 
  1396 #if defined(__WINS__)
  1397 void TWsPointer::SetXyInputType(TXYInputType aXyInputType)
  1398 	{
  1399 	if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse)
  1400 		{
  1401 		// change from Mouse types to Pointer/None
  1402 		for (TInt ii = 0; ii < iMaxPointers; ii++)
  1403 			{
  1404 			if (iPointers[ii].iState != EPointerStateDown)
  1405 				{
  1406 				iPointers[ii].iCurrentWindow=NULL;
  1407 				}
  1408 			}
  1409 		UpdatePointerCursor();
  1410 		}
  1411 	else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer)
  1412 		{
  1413 		// change from Pointer/None types to Mouse types
  1414 		for (TInt ii = 0; ii < iMaxPointers; ii++)
  1415 			{
  1416 			if (iPointers[ii].iState != EPointerStateDown)
  1417 				{
  1418 				TPoint pos(iPointers[ii].iPos);
  1419 				TPoint parPos;
  1420 				iPointers[ii].ReLogCurrentWindow(pos,parPos,NULL);
  1421 				}
  1422 			}
  1423 		UpdatePointerCursor();
  1424 		}
  1425 	iXyInputType=aXyInputType;
  1426 	}
  1427 #endif
  1428 
  1429 /**
  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.
  1432 
  1433 This method implements single pointer environment emulation rules (see design
  1434 documentation for more details).
  1435 
  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).
  1438 */
  1439 void TWsPointer::UpdatePrimaryPointer(const TRawEvent& aRawEvent)
  1440 	{
  1441 	iPreviousPrimaryPointer = iPrimaryPointer;
  1442 	
  1443 	TRawEvent::TType type=aRawEvent.Type();
  1444 	TInt pointerNumber = aRawEvent.PointerNumber();
  1445     
  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++)
  1449 		{
  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);
  1451 		}
  1452 #endif
  1453 	
  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)
  1457 		{
  1458 		if (type != TRawEvent::EPointer3DOutOfRange && iPointers[pointerNumber].iState == EPointerStateOutOfRange)
  1459 			{
  1460 			iPrimaryPointer = pointerNumber;
  1461 #ifdef LOG_WSERV_EVENTS
  1462 	        RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(OutRange) = %d",iPrimaryPointer);
  1463 #endif
  1464 			}
  1465 		return;
  1466 		}
  1467 	
  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)
  1472 		{
  1473 		iPrimaryPointer = pointerNumber;
  1474 #ifdef LOG_WSERV_EVENTS
  1475         RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(ButtonDown) = %d",iPrimaryPointer);
  1476 #endif
  1477 		return;
  1478 		}
  1479 	}
  1480 
  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
  1487 */
  1488 TInt TWsPointer::SetCloseProximityThresholds(TInt aEnterCloseProximityThreshold, TInt aExitCloseProximityThreshold)
  1489 	{
  1490 	if (aEnterCloseProximityThreshold < aExitCloseProximityThreshold)
  1491 		{
  1492 		return KErrArgument;
  1493 		}
  1494 	
  1495 	TInt ret = HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, aEnterCloseProximityThreshold);
  1496 	if (ret != KErrNone)
  1497 		{
  1498 		return ret;
  1499 		}
  1500 	
  1501 	ret = HAL::Set(HALData::EPointer3DExitCloseProximityThreshold, aExitCloseProximityThreshold);
  1502 	WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicProxThresholdsInconsist);
  1503 	if (ret != KErrNone)
  1504 		{
  1505 		HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, iEnterCloseProximityThreshold);
  1506 		return ret;
  1507 		}
  1508 	
  1509 	iEnterCloseProximityThreshold = aEnterCloseProximityThreshold;
  1510 	iExitCloseProximityThreshold  = aExitCloseProximityThreshold; 
  1511 	return KErrNone;
  1512 	}
  1513 
  1514 /**
  1515 @return Z coordinate threshold value for TPointerEvent::EEnterCloseProximity events
  1516 @see RWsSession::GetEnterCloseProximityThreshold which calls this method
  1517 */
  1518 TInt TWsPointer::GetEnterCloseProximityThreshold()
  1519 	{
  1520 	return iEnterCloseProximityThreshold;
  1521 	}
  1522 
  1523 /**
  1524 @return Z coordinate threshold value for TPointerEvent::EExitCloseProximity events
  1525 @see RWsSession::GetExitCloseProximityThreshold which calls this method
  1526 */
  1527 TInt TWsPointer::GetExitCloseProximityThreshold()
  1528 	{
  1529 	return iExitCloseProximityThreshold;
  1530 	}
  1531 
  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
  1538 */
  1539 TInt TWsPointer::SetHighPressureThresholds(TInt aEnterHighPressureThreshold, TInt aExitHighPressureThreshold)
  1540 	{
  1541 	if (aEnterHighPressureThreshold < aExitHighPressureThreshold)
  1542 		{
  1543 		return KErrArgument;
  1544 		}
  1545 	
  1546 	TInt ret = HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, aEnterHighPressureThreshold);
  1547 	if (ret != KErrNone)
  1548 		{
  1549 		return ret;
  1550 		}
  1551 	
  1552 	ret = HAL::Set(HALData::EPointer3DExitHighPressureThreshold, aExitHighPressureThreshold);
  1553 	WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicPressThresholdsInconsist);
  1554 	if (ret != KErrNone)
  1555 		{
  1556 		HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, iEnterHighPressureThreshold);
  1557 		return ret;
  1558 		}
  1559 	
  1560 	iEnterHighPressureThreshold = aEnterHighPressureThreshold;
  1561 	iExitHighPressureThreshold  = aExitHighPressureThreshold;
  1562 	return KErrNone;
  1563 	}
  1564 
  1565 /**
  1566 @return Z coordinate threshold value for TPointerEvent::EEnterHighPressure events
  1567 @see RWsSession::GetEnterHighPressureThreshold which calls this method
  1568 */
  1569 TInt TWsPointer::GetEnterHighPressureThreshold()
  1570 	{
  1571 	return iEnterHighPressureThreshold;
  1572 	}
  1573 
  1574 /**
  1575 @return Z coordinate threshold value for TPointerEvent::EExitHighPressure events
  1576 @see RWsSession::GetExitHighPressureThreshold which calls this method
  1577 */
  1578 TInt TWsPointer::GetExitHighPressureThreshold()
  1579 	{
  1580 	return iExitHighPressureThreshold;
  1581 	}
  1582 
  1583 
  1584 /**
  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.
  1589 
  1590 @param aY Current y coordinate pointer position.
  1591  
  1592  */
  1593 void TWsPointer::ShiftYCoordinate(TInt& aY)
  1594     {
  1595     WS_ASSERT_DEBUG(iYOffset>=0, EWsPanicInvalidPointerOffset);
  1596     if (aY >= iYOffset)
  1597         {
  1598         aY -=iYOffset;
  1599         }
  1600     else
  1601         {
  1602         aY=0;
  1603         }
  1604     }
  1605 //
  1606 CWsPointerTimer::CWsPointerTimer(MPointerTimerCallback& aPointerTimerCallback)
  1607 : CTimer(EPointerRepeatPriority), iPointerTimerCallback(aPointerTimerCallback)
  1608 	{}
  1609 
  1610 void CWsPointerTimer::ConstructL()
  1611 	{
  1612 	CTimer::ConstructL();
  1613 	CActiveScheduler::Add(this);
  1614 	}
  1615 
  1616 CWsPointerTimer* CWsPointerTimer::NewL(MPointerTimerCallback& aWsPointer)
  1617 	{
  1618 	CWsPointerTimer* self = new(ELeave) CWsPointerTimer(aWsPointer);
  1619 	CleanupStack::PushL(self);
  1620 	self->ConstructL();
  1621 	CleanupStack::Pop(self);
  1622 	return self;
  1623 	}
  1624 
  1625 void CWsPointerTimer::RunL()
  1626 	{
  1627 	User::ResetInactivityTime();
  1628 	WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
  1629 	iPointerTimerCallback.RepeatTimerCompleted();
  1630 	}
  1631 //
  1632 
  1633 CWsPointerBuffer::~CWsPointerBuffer()
  1634 	{
  1635 	if (this == iCurrentBuffer)
  1636 		{
  1637 		// We're about to be destroyed - don't want to be pointed at any more.
  1638 		iCurrentBuffer = NULL;
  1639 		}
  1640 	iList.Remove(*this);
  1641 	}
  1642 
  1643 void CWsPointerBuffer::ConnectL(CWsClientWindow* aWindow, TInt aMaxPoints, TUint aFlags)
  1644 	{
  1645 	CWsPointerBuffer* pb=NULL;
  1646 	for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
  1647 		{
  1648 		if (pb->iWindow==aWindow)
  1649 			{
  1650 			User::Leave(KErrInUse);
  1651 			}
  1652 		}
  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);
  1659 	AdjustMaxSizeL();
  1660 	CleanupStack::Pop();
  1661 	}
  1662 
  1663 void CWsPointerBuffer::Disconnect(CWsClientWindow* aWindow)
  1664 	{
  1665 	CWsPointerBuffer* pb=NULL;
  1666 	for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
  1667 		{
  1668 		if (pb->iWindow==aWindow)
  1669 			{
  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
  1673 			}
  1674 		}
  1675 	}
  1676 
  1677 void CWsPointerBuffer::Reset()
  1678 	{
  1679 	iSignalled=EFalse;
  1680 	iPointerBuffer->Reset();
  1681 	}
  1682 
  1683 void CWsPointerBuffer::SignalBufferReady()
  1684 	{
  1685 	if (!iSignalled)
  1686 		{
  1687 		if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
  1688 			{
  1689 			iSignalled=ETrue;
  1690 			}
  1691 		}
  1692 	}
  1693 
  1694 void CWsPointerBuffer::PointerEvent(CWsClientWindow* aWindow,const TPoint &aPoint)
  1695 	{
  1696 	if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
  1697 		{
  1698 		Reset();
  1699 		CWsPointerBuffer* pb=NULL;
  1700 		for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
  1701 			{
  1702 			if (pb->iWindow==aWindow)
  1703 				{
  1704 				iCurrentBuffer=pb;
  1705 				break; // from for loop
  1706 				}
  1707 			}
  1708 		}
  1709 	iPointerBuffer->Add(&aPoint);
  1710 	SignalBufferReady();
  1711 	}
  1712 
  1713 void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow* aWindow,TInt aMaxPoints)
  1714 	{
  1715 	enum {KPointerMoveBufferSize=32};	// Puts 256 bytes on the stack
  1716 	if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
  1717 		{
  1718 		iSignalled=EFalse;
  1719 		TInt max=Min(aMaxPoints,iPointerBuffer->Count());
  1720 		TInt buflen=0;
  1721 		aWindow->WsOwner()->SetReply(max);
  1722 		TPoint point;
  1723 		TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
  1724 		for(TInt index=0;index<max;index++)
  1725 			{
  1726 			iPointerBuffer->Remove(&point);
  1727 			pnts.Append((TUint8 *)&point,sizeof(TPoint));
  1728 			buflen++;
  1729 			if (buflen==KPointerMoveBufferSize)
  1730 				{
  1731 				CWsClient::ReplyBuf(pnts);
  1732 				pnts.Zero();
  1733 				buflen=0;
  1734 				}
  1735 			}
  1736 		if (buflen>0)
  1737 			{
  1738 			CWsClient::ReplyBuf(pnts);
  1739 			}
  1740 		if (iPointerBuffer->Count())
  1741 			{
  1742 			SignalBufferReady();
  1743 			}
  1744 		}
  1745 	}
  1746 
  1747 void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
  1748 	{
  1749 	if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
  1750 		{
  1751 		Reset();
  1752 		}
  1753 	}
  1754 
  1755 void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow* aWindow)
  1756 	{
  1757 	if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
  1758 		{
  1759 		Reset();
  1760 		}
  1761 	}
  1762 
  1763 void CWsPointerBuffer::AdjustMaxSizeL()
  1764 	{
  1765 	TInt max=0;
  1766 	CWsPointerBuffer* pb=NULL;
  1767 	for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
  1768 		{
  1769 		if (pb->iMaxPoints>max)
  1770 			{
  1771 			max=pb->iMaxPoints;
  1772 			}
  1773 		}
  1774 	if (max==0)
  1775 		{
  1776 		delete iPointerBuffer;
  1777 		iPointerBuffer=NULL;
  1778 		}
  1779 	else if (!iPointerBuffer)
  1780 		{
  1781 		CCirBuf<TPoint>* pointerBuffer=new(ELeave) CCirBuf<TPoint>;
  1782 		CleanupStack::PushL(pointerBuffer);
  1783 		pointerBuffer->SetLengthL(max);
  1784 		CleanupStack::Pop();
  1785 		iPointerBuffer=pointerBuffer;
  1786 		}
  1787 	else
  1788 		{
  1789 		iPointerBuffer->SetLengthL(max);
  1790 		}
  1791 	}