1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/POINTER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1791 @@
1.4 +// Copyright (c) 1995-2010 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// Pointer functions
1.18 +//
1.19 +//
1.20 +
1.21 +#include <e32std.h>
1.22 +#include <e32hal.h>
1.23 +#include "W32CLICK.H"
1.24 +#include "pointer.h"
1.25 +#include "rootwin.h"
1.26 +#include "windowgroup.h"
1.27 +#include "KEYCLICK.H"
1.28 +#include "ScrDev.H"
1.29 +#include "EVENT.H"
1.30 +#include "panics.h"
1.31 +#include "wstop.h"
1.32 +#include "inifile.h"
1.33 +#include <hal.h>
1.34 +#include "advancedpointereventhelper.h"
1.35 +#include "graphics/pointereventdata.h"
1.36 +#include "debughelper.h"
1.37 +
1.38 +TTimeIntervalMicroSeconds32 TWsPointer::iDoubleClickMaxInterval;
1.39 +TInt TWsPointer::iDoubleClickMaxDistance;
1.40 +CWsPointerCursor* TWsPointer::iCursorSprite;
1.41 +TPointerCursorMode TWsPointer::iPointerCursorMode=EPointerCursorNormal;
1.42 +TXYInputType TWsPointer::iXyInputType;
1.43 +TBool TWsPointer::iTimerQueued;
1.44 +TBool TWsPointer::iUpdateRequired;
1.45 +CPeriodic* TWsPointer::iPeriodicTimer;
1.46 +CWsRootWindow* TWsPointer::iRootWindow;
1.47 +TInt TWsPointer::iMaxPointers;
1.48 +TBool TWsPointer::iIs3DPointer;
1.49 +RArray<TWsPointer> TWsPointer::iPointers;
1.50 +TInt TWsPointer::iPrimaryPointer = TAdvancedPointerEvent::EDefaultPointerNumber;
1.51 +TInt TWsPointer::iPreviousPrimaryPointer;
1.52 +TInt TWsPointer::iEnterCloseProximityThreshold;
1.53 +TInt TWsPointer::iExitCloseProximityThreshold;
1.54 +TInt TWsPointer::iEnterHighPressureThreshold;
1.55 +TInt TWsPointer::iExitHighPressureThreshold;
1.56 +TBool CWsPointerBuffer::iSignalled=EFalse;
1.57 +CWsPointerBuffer* CWsPointerBuffer::iCurrentBuffer=NULL;
1.58 +CCirBuf<TPoint>* CWsPointerBuffer::iPointerBuffer=NULL;
1.59 +TSglQue<CWsPointerBuffer> CWsPointerBuffer::iList(_FOFF(CWsPointerBuffer,iQue));
1.60 +TInt TWsPointer::iYOffset;
1.61 +#if defined(__WINS__)
1.62 +TBool TWsPointer::iEmulatorRotatePointerCoords;
1.63 +#endif
1.64 +
1.65 +static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_SwEvent,ECapabilitySwEvent);
1.66 +
1.67 +void TWsPointer::InitStaticsL()
1.68 + {
1.69 + //This iYOffset setting is specific for capacitive touch screens, where user's finger is the pointer device.
1.70 + //This is typically used so that the pointer event location is more inline with where the user perceives their
1.71 + //finger to be on the screen (for example, due to refraction and the relatively large touch area of a finger).
1.72 + iYOffset = 0;
1.73 + _LIT( KWSERVIniFileVarYShifting, "YSHIFTING");
1.74 + TBool fetchingSucceeded = WsIniFile->FindVar(KWSERVIniFileVarYShifting, iYOffset);
1.75 + WS_ASSERT_ALWAYS(iYOffset>=0, EWsPanicInvalidPointerOffset);
1.76 + if ( !fetchingSucceeded )
1.77 + {
1.78 + iYOffset = 0;
1.79 + }
1.80 +
1.81 +#if defined(__WINS__)
1.82 + //An emulator may or may not deploy a renderchain or displaydriver that supports rotated drawing.
1.83 + //On a real device target the coordinate system is always rotated together with wserv's screendevice.
1.84 + _LIT( KWSERVIniFileVarEmulatorRotPointCoords, "EMULATOR_ROTATE_POINTER_COORDS");
1.85 + iEmulatorRotatePointerCoords = WsIniFile->FindVar(KWSERVIniFileVarEmulatorRotPointCoords);
1.86 +#endif
1.87 +
1.88 + const CScreen* screen = CWsTop::Screen();
1.89 + WS_ASSERT_ALWAYS(screen, EWsPanicNoScreen);
1.90 + iRootWindow = screen->RootWindow();
1.91 +
1.92 + TMachineInfoV1Buf machineInfo;
1.93 + UserHal::MachineInfo(machineInfo);
1.94 + iXyInputType=machineInfo().iXYInputType;
1.95 +
1.96 + // Read EPointerMaxPointers from HAL and check if reported value is consistent
1.97 + // with iXyInputType.
1.98 + // Even if HAL reports that device doesn't support any pointer, WSERV
1.99 + // always has to support at least one pointer for compatibility reasons
1.100 + // and for keeping state of pointer cursor.
1.101 + if(HAL::Get(HALData::EPointerMaxPointers,iMaxPointers)!=KErrNone)
1.102 + {
1.103 + iMaxPointers = 1;
1.104 + }
1.105 + else
1.106 + {
1.107 + WS_ASSERT_ALWAYS(iMaxPointers >= 0 && iMaxPointers <= TAdvancedPointerEvent::EMaximumWServNumberOfPointers,
1.108 + EWsPanicMaxPointersOutOfRange);
1.109 + WS_ASSERT_ALWAYS(( XyInput() && iMaxPointers > 0) || (!XyInput() && iMaxPointers == 0),
1.110 + EWsPanicMaxPointersInconsistent);
1.111 + if (iMaxPointers == 0)
1.112 + {
1.113 + iMaxPointers = 1;
1.114 + }
1.115 + }
1.116 +
1.117 + //** Log the number of pointers here i,e iMaxPointers
1.118 +#ifdef LOG_WSERV_EVENTS
1.119 + RDebug::Printf("_WSEVENT_POINTER: Number of pointers system supports %d", iMaxPointers);
1.120 +#endif
1.121 +
1.122 + // Does device support Z coordinate of the pointers?
1.123 + if(HAL::Get(HALData::EPointer3D,iIs3DPointer)!=KErrNone)
1.124 + {
1.125 + iIs3DPointer=EFalse; // No API, then no 3D pointers
1.126 + }
1.127 + WS_ASSERT_ALWAYS(!iIs3DPointer || XyInput(), EWsPanicPointer3DInconsistent);
1.128 +
1.129 +#ifdef LOG_WSERV_EVENTS
1.130 + RDebug::Printf("_WSEVENT_POINTER: Z coordinate supported %d", iIs3DPointer);
1.131 +#endif
1.132 +
1.133 + // Initialize thresholds for EEnterCloseProximity, EExitCloseProximity,
1.134 + // EEnterHighPressure and EExitHightPressure events.
1.135 + if(HAL::Get(HALData::EPointer3DEnterCloseProximityThreshold,
1.136 + iEnterCloseProximityThreshold) != KErrNone)
1.137 + {
1.138 + iEnterCloseProximityThreshold = KMaxTInt;
1.139 + }
1.140 + if(HAL::Get(HALData::EPointer3DExitCloseProximityThreshold,
1.141 + iExitCloseProximityThreshold) != KErrNone)
1.142 + {
1.143 + iExitCloseProximityThreshold = KMinTInt;
1.144 + }
1.145 + if(HAL::Get(HALData::EPointer3DEnterHighPressureThreshold,
1.146 + iEnterHighPressureThreshold) != KErrNone)
1.147 + {
1.148 + iEnterHighPressureThreshold = KMaxTInt;
1.149 + }
1.150 + if(HAL::Get(HALData::EPointer3DExitHighPressureThreshold,
1.151 + iExitHighPressureThreshold) != KErrNone)
1.152 + {
1.153 + iExitHighPressureThreshold = KMinTInt;
1.154 + }
1.155 +
1.156 + iPointers = RArray<TWsPointer>(iMaxPointers);
1.157 + TWsPointer emptyPointer;
1.158 + emptyPointer.iRepeatTimer = NULL;
1.159 + emptyPointer.Clear();
1.160 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.161 + {
1.162 + emptyPointer.iNumber = ii;
1.163 + User::LeaveIfError(iPointers.Append(emptyPointer));
1.164 + RepeatTimer(ii) = CWsPointerTimer::NewL(iPointers[ii]);
1.165 + }
1.166 +
1.167 + iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
1.168 + }
1.169 +
1.170 +void TWsPointer::Clear()
1.171 + {
1.172 + iState = EPointerStateOutOfRange;
1.173 + iPos.iX = 0;
1.174 + iPos.iY = 0;
1.175 + iPressureProximity = 0;
1.176 + iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
1.177 + iActualWinPointerIsOver = NULL;
1.178 + iGrabWindow = NULL;
1.179 + iLastUnmatchedDown1 = NULL;
1.180 + iLastUnmatchedDown2 = NULL;
1.181 + iLastUnmatchedDown3 = NULL;
1.182 + iLastUnmatchedEnterHighPressure = NULL;
1.183 + iPrevClickWindow = NULL;
1.184 + iInCloseProximity = EFalse;
1.185 + iInHighPressure = EFalse;
1.186 + CancelPointerRepeatEventRequest();
1.187 + }
1.188 +
1.189 +/**
1.190 +Turns off pointer cursor, deletes permenently pointer cursor update timer and pointer event repeat timer.
1.191 +*/
1.192 +void TWsPointer::Stop()
1.193 + {
1.194 + SetPointerCursorMode(EPointerCursorNone);
1.195 + UpdatePointerCursor();
1.196 + delete iPeriodicTimer;
1.197 + }
1.198 +
1.199 +void TWsPointer::DeleteStatics()
1.200 + {
1.201 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.202 + {
1.203 + delete RepeatTimer(ii);
1.204 + }
1.205 + iPointers.Close();
1.206 + }
1.207 +
1.208 +void TWsPointer::SetPointerCursorPos(TPoint aPos)
1.209 + {
1.210 + RestrictPos(aPos,EFalse);
1.211 + iPointers[iPrimaryPointer].iPos=aPos;
1.212 + iPointers[iPrimaryPointer].ReLogCurrentWindow();
1.213 + UpdatePointerCursor();
1.214 + }
1.215 +
1.216 +void TWsPointer::SendEnterExitEvent(TEventCode aType)
1.217 + {
1.218 + if (iCurrentWindow
1.219 + && (iState != EPointerStateOutOfRange)
1.220 + && (iNumber == iPrimaryPointer || iCurrentWindow->AdvancedPointersEnabled())
1.221 + && !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
1.222 + && !iCurrentWindow->ShutDownInProgress())
1.223 + {
1.224 + iCurrentWindow->QueueEvent(aType, iNumber);
1.225 + }
1.226 + }
1.227 +
1.228 +void TWsPointer::SetCurrentWindow(const CWsWindow* aWin)
1.229 + {
1.230 + if (aWin!=iCurrentWindow)
1.231 + {
1.232 + SendEnterExitEvent(EEventPointerExit);
1.233 + iCurrentWindow=aWin;
1.234 + SendEnterExitEvent(EEventPointerEnter);
1.235 + }
1.236 + }
1.237 +
1.238 +/**
1.239 +Relogs the current window (sets iCurrentWindow) for this pointer.
1.240 +Retrieves pointer relative position in current window and pointer position in current
1.241 +window's parent window.
1.242 +*/
1.243 +void TWsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup* aForceInGroup)
1.244 + {
1.245 + if (iRootWindow)
1.246 + {
1.247 + SetCurrentWindow(iRootWindow->PointerWindow(iPos,&aPos,&aParentPos,iGrabWindow,
1.248 + iActualWinPointerIsOver,aForceInGroup));
1.249 + }
1.250 + else
1.251 + {
1.252 + iCurrentWindow=NULL;
1.253 + }
1.254 + }
1.255 +
1.256 +/**
1.257 +Relog all pointer's current windows when the window layout has changed.
1.258 +*/
1.259 +void TWsPointer::ReLogPointersCurrentWindows()
1.260 + {
1.261 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.262 + {
1.263 + TWsPointer& pointer = iPointers[ii];
1.264 + pointer.ReLogCurrentWindow();
1.265 + }
1.266 + }
1.267 +
1.268 +/*
1.269 +Relog this pointer's current pointer window when the window layout has changed.
1.270 +Works similarly to ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup* aForceInGroup),
1.271 +but doesn't set relative positions.
1.272 +*/
1.273 +void TWsPointer::ReLogCurrentWindow()
1.274 + {
1.275 + if (iCurrentWindow)
1.276 + {
1.277 + SetCurrentWindow(iRootWindow->PointerWindow(iPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
1.278 + }
1.279 + }
1.280 +
1.281 +/*
1.282 +Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
1.283 +is the current window
1.284 +*/
1.285 +void TWsPointer::ReLogWindow(const CWsWindow* aWin)
1.286 + {
1.287 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.288 + {
1.289 + if (aWin == iPointers[ii].iCurrentWindow)
1.290 + {
1.291 + iPointers[ii].SendEnterExitEvent(EEventPointerEnter);
1.292 + }
1.293 + }
1.294 + }
1.295 +
1.296 +void TWsPointer::UnmatchedEventPurged(TPointerEvent::TType aPointerType, TUint aHandle)
1.297 + {
1.298 + if (aPointerType==TPointerEvent::EButton1Up)
1.299 + {
1.300 + if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
1.301 + {
1.302 + iGrabWindow=NULL;
1.303 + }
1.304 + if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
1.305 + {
1.306 + CancelPointerRepeatEventRequest();
1.307 + }
1.308 + }
1.309 + switch(aPointerType)
1.310 + {
1.311 + case TPointerEvent::EButton1Up:
1.312 + iLastUnmatchedDown1=aHandle;
1.313 + break;
1.314 + case TPointerEvent::EButton2Up:
1.315 + iLastUnmatchedDown2=aHandle;
1.316 + break;
1.317 + case TPointerEvent::EButton3Up:
1.318 + iLastUnmatchedDown3=aHandle;
1.319 + break;
1.320 + case TPointerEvent::EExitHighPressure:
1.321 + iLastUnmatchedEnterHighPressure=aHandle;
1.322 + break;
1.323 + default:;
1.324 + }
1.325 + }
1.326 +
1.327 +void TWsPointer::WindowDisconnected(const CWsWindow* deletedWindow)
1.328 + {
1.329 + for (TInt pointerNum = 0; pointerNum < iMaxPointers; pointerNum++)
1.330 + {
1.331 + TWsPointer& pointer = iPointers[pointerNum];
1.332 + if (pointer.iRepeatWindow==deletedWindow)
1.333 + {
1.334 + pointer.CancelPointerRepeatEventRequest();
1.335 + }
1.336 + if (pointer.iGrabWindow==deletedWindow)
1.337 + {
1.338 + pointer.iGrabWindow=NULL;
1.339 + }
1.340 + if (pointer.iCurrentWindow==deletedWindow)
1.341 + {
1.342 + pointer.ReLogCurrentWindow();
1.343 + if (pointerNum == iPrimaryPointer)
1.344 + {
1.345 + UpdatePointerCursor();
1.346 + }
1.347 + }
1.348 + }
1.349 + }
1.350 +
1.351 +/*
1.352 +Callback function pointer for up event remove event queue walk
1.353 +*/
1.354 +TEventQueueWalkRet RemovePointerUpFunc(TAny* aHandle, TWsEvent* aEvent)
1.355 + {
1.356 + if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
1.357 + {
1.358 + return(EEventQueueWalkDeleteEvent);
1.359 + }
1.360 + return(EEventQueueWalkOk);
1.361 + }
1.362 +
1.363 +/*
1.364 +If the specified pointer is down claim grab in aWindow as though the down event had
1.365 +gone to this window. Also send an up event to the window (if any) that would receive it if the
1.366 +pointer was released now.
1.367 +If no pointer is specifed do the above for the primary pointer.
1.368 +
1.369 +@return KErrNone if successful,
1.370 + KErrNotFound if pointernumber out of range,
1.371 + KErrNotSupported if incorrect pointer grab claimed for window in emulation mode,
1.372 + KErrPermissionDenied if trying to grab from a different window owner without the required capability.
1.373 +*/
1.374 +TInt TWsPointer::ClaimGrab(const CWsWindow *aWindow,const TWsWinCmdGrabControl& aGrabControl)
1.375 + {
1.376 + TInt pointerNumber(aGrabControl.HasPointerNumber() ? aGrabControl.pointerNumber : iPrimaryPointer);
1.377 + TBool advancedPointersEnabled(aWindow->AdvancedPointersEnabled());
1.378 + TInt errNo(KErrNone);
1.379 +
1.380 + if(!advancedPointersEnabled && aGrabControl.HasPointerNumber() && (TAdvancedPointerEvent::EDefaultPointerNumber!=pointerNumber))
1.381 + {
1.382 + // The window is in emulation mode, and cannot get events for anything other than the primary pointer
1.383 + errNo=KErrNotSupported;
1.384 + }
1.385 + else if(advancedPointersEnabled && ((pointerNumber<0) || (pointerNumber>=iMaxPointers)))
1.386 + {
1.387 + // Unknown pointer number
1.388 + errNo=KErrNotFound;
1.389 + }
1.390 + else if(aGrabControl.HasPointerNumber() && (aWindow->WsOwner()!=iPointers[pointerNumber].iCurrentWindow->WsOwner())
1.391 + && !KSecurityPolicy_SwEvent().CheckPolicy(aWindow->WsOwner()->ClientMessage(), __PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowBase::ClaimPointerGrab")))
1.392 + {
1.393 + // Trying to grab from a different window owner requires the relevant permission
1.394 + // - Only for the multipointer API, RWindowBase::ClaimPointerGrab(TInt, TBool), which provides a pointer number
1.395 + // - Doing so for the legacy non-multipointer API, RWindowBase::ClaimPointerGrab(TBool), would be a compatibility break
1.396 + errNo=KErrPermissionDenied;
1.397 + }
1.398 + else
1.399 + {
1.400 + iPointers[pointerNumber].ClaimGrab(aWindow,aGrabControl.CheckFlags(TWsWinCmdGrabControl::ESendUpEvent));
1.401 + }
1.402 +
1.403 + return errNo;
1.404 + }
1.405 +
1.406 +/*
1.407 +If this pointer is down claim grab in aWindow as though the down event had
1.408 +gone to this window. Also send an up event to the window (if any) that would receive it if the
1.409 +pointer was released now.
1.410 +*/
1.411 +void TWsPointer::ClaimGrab(const CWsWindow* aWindow,TBool aSendUpEvent)
1.412 + {
1.413 + TInt modState=TWindowServerEvent::GetModifierState();
1.414 + TWsEvent event;
1.415 + TAdvancedPointerEvent& pointerEvent=*event.Pointer();
1.416 + TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event, TPointerEvent::EButton1Up,modState,TPoint3D(iPos.iX,iPos.iY,iPressureProximity),iNumber);
1.417 + if (iState == EPointerStateDown)
1.418 + {
1.419 + if (iCurrentWindow!=aWindow)
1.420 + {
1.421 + if (aSendUpEvent)
1.422 + {
1.423 + ProcessEvent(event,NULL,EFalse);
1.424 + }
1.425 + else // If up event already in queue purge it
1.426 + {
1.427 + CEventQueue* eventQueue = iCurrentWindow->EventQueue();
1.428 + if (eventQueue)
1.429 + {
1.430 + TUint handle=iCurrentWindow->ClientHandle();
1.431 + eventQueue->WalkEventQueue(&RemovePointerUpFunc,&handle);
1.432 + }
1.433 + }
1.434 + iState = EPointerStateDown;
1.435 + if (aWindow->HasPointerGrab())
1.436 + {
1.437 + iGrabWindow=aWindow;
1.438 + }
1.439 + ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
1.440 + pointerEvent.iType=TPointerEvent::EDrag;
1.441 + ProcessPointerEvent(event);
1.442 + }
1.443 + }
1.444 + else if (iState == EPointerStateUp)
1.445 + {
1.446 + const CWsWindow *current=iCurrentWindow;
1.447 + iCurrentWindow=aWindow;
1.448 + WS_ASSERT_DEBUG(iGrabWindow==NULL, EWsPanicPointerClaimGrab);
1.449 + iGrabWindow=aWindow; // Force the up event to be sent to aWindow
1.450 + ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
1.451 + ProcessPointerEvent(event);
1.452 + iGrabWindow=NULL;
1.453 + iCurrentWindow=current;
1.454 + }
1.455 + }
1.456 +
1.457 +/**
1.458 +@return ETrue if matching event has been purged, so current event should not be delivered,
1.459 + EFalse otherwise.
1.460 +*/
1.461 +TBool TWsPointer::CheckMatchingEventPurged(TPointerEvent::TType aType)
1.462 + {
1.463 + switch(aType)
1.464 + {
1.465 + TUint lastUnmatchedDown;
1.466 + case TPointerEvent::EButton1Up:
1.467 + lastUnmatchedDown=iLastUnmatchedDown1;
1.468 + iLastUnmatchedDown1=0;
1.469 + iLastUnmatchedEnterHighPressure=0;
1.470 + goto lastUnmatchedDownCheck;
1.471 + case TPointerEvent::EButton2Up:
1.472 + lastUnmatchedDown=iLastUnmatchedDown2;
1.473 + iLastUnmatchedDown2=0;
1.474 + goto lastUnmatchedDownCheck;
1.475 + case TPointerEvent::EButton3Up:
1.476 + lastUnmatchedDown=iLastUnmatchedDown3;
1.477 + iLastUnmatchedDown3=0;
1.478 + goto lastUnmatchedDownCheck;
1.479 + case TPointerEvent::EExitHighPressure:
1.480 + lastUnmatchedDown=iLastUnmatchedEnterHighPressure;
1.481 + iLastUnmatchedEnterHighPressure=0;
1.482 +lastUnmatchedDownCheck:
1.483 + if (lastUnmatchedDown==iCurrentWindow->ClientHandle())
1.484 + {
1.485 + return ETrue; // Don't deliver the event as we've already thrown away matching event
1.486 + }
1.487 + default: //Should never get to default
1.488 + break;
1.489 + }
1.490 + return EFalse;
1.491 + }
1.492 +
1.493 +TBool TWsPointer::QueuePointerEvent(const CWsWindow* aWindow, TWsEvent &aEvent)
1.494 + {
1.495 + if (aWindow->WsOwner() &&
1.496 + (aWindow->AdvancedPointersEnabled() ||
1.497 + TAdvancedPointerEventHelper::PointerNumber(aEvent) == iPrimaryPointer))
1.498 + {
1.499 + CEventQueue* queue=aWindow->EventQueue();
1.500 + aEvent.SetHandle(aWindow->ClientHandle());
1.501 +#ifdef LOG_WSERV_EVENTS
1.502 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent with AdvancedPointerEnabled");
1.503 +#endif
1.504 + if (aEvent.Handle()!=0)
1.505 + {
1.506 + if(!aWindow->AdvancedPointersEnabled())
1.507 + {
1.508 + // Re-assign from WServ primary pointer number, to EDefaultPointerNumber for Cone and other clients.
1.509 + // This should not get confused with any real pointer of the same number due to the TWsEvents' window handle
1.510 + // being different.
1.511 + TAdvancedPointerEventHelper::SetPointerNumber(aEvent,TAdvancedPointerEvent::EDefaultPointerNumber);
1.512 + aEvent.Pointer()->iModifiers&=~EModifierAdvancedPointerEvent; // Clear the advanced pointer flag
1.513 + }
1.514 + if (queue->UpdateLastPointerEvent(aEvent))
1.515 + {
1.516 + return EFalse;
1.517 + }
1.518 + TWservEventPriorities priority=EEventPriorityLow;
1.519 + switch (aEvent.Pointer()->iType)
1.520 + {
1.521 + case TPointerEvent::EButton1Up:
1.522 + case TPointerEvent::EButton2Up:
1.523 + case TPointerEvent::EButton3Up:
1.524 + case TPointerEvent::EExitHighPressure:
1.525 + if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
1.526 + {
1.527 + #ifdef LOG_WSERV_EVENTS
1.528 + RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 01");
1.529 + #endif
1.530 + return ETrue;
1.531 + }
1.532 + if (queue->CheckRoom())
1.533 + {
1.534 + if (CheckMatchingEventPurged(aEvent.Pointer()->iType))
1.535 + {
1.536 + #ifdef LOG_WSERV_EVENTS
1.537 + RDebug::Printf("_WSEVENT_POINTER: Check matching event has been purged so no addition of event 02");
1.538 + #endif
1.539 + return ETrue;
1.540 + }
1.541 + }
1.542 + /*Fall Through if an event was not purged*/
1.543 + case TPointerEvent::EButton1Down:
1.544 + case TPointerEvent::EButton2Down:
1.545 + case TPointerEvent::EButton3Down:
1.546 + case TPointerEvent::EEnterHighPressure:
1.547 + case TPointerEvent::EOutOfRange:
1.548 + priority=EEventPriorityHigh;
1.549 + break;
1.550 + default:;
1.551 + }
1.552 +#ifdef LOG_WSERV_EVENTS
1.553 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::QueuePointerEvent After adding event to clientqueue Event State %d ", iState);
1.554 +#endif
1.555 + queue->QueueEvent(aEvent,priority);
1.556 + }
1.557 + }
1.558 + return EFalse;
1.559 + }
1.560 +
1.561 +/*
1.562 +Moves the window group which contains this pointer's iCurrentWindow on top.
1.563 +*/
1.564 +void TWsPointer::ProcessForegroundCheck()
1.565 + {
1.566 + CWsWindowGroup* group=((CWsTopClientWindow *)iCurrentWindow)->TopClientWindow()->Parent();
1.567 + if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
1.568 + {
1.569 + group->SetOrdinalPosition(0);
1.570 + }
1.571 + }
1.572 +
1.573 +/*
1.574 +Pointer Event Processing - stage 3 of 3:
1.575 +- setting event's time
1.576 +- auto foreground check
1.577 +- double clicks detection
1.578 +- pointer repeats
1.579 +- add event to client's queue
1.580 +- drag&drop capturing
1.581 +*/
1.582 +void TWsPointer::ProcessPointerEvent(TWsEvent& aEvent)
1.583 + {
1.584 + if (iCurrentWindow && iCurrentWindow!=iRootWindow)
1.585 + {
1.586 + aEvent.SetType(EEventPointer);
1.587 + aEvent.SetTimeNow();
1.588 + TPointerEvent::TType type=aEvent.Pointer()->iType;
1.589 + switch(type)
1.590 + {
1.591 + //TUint lastUnmatchedDown;
1.592 + case TPointerEvent::EButton1Down:
1.593 + ProcessForegroundCheck();
1.594 + /*Fall Through*/
1.595 + case TPointerEvent::EButton2Down:
1.596 + case TPointerEvent::EButton3Down:
1.597 + {
1.598 + TPoint& pos=aEvent.Pointer()->iPosition;
1.599 + if (iCurrentWindow==iPrevClickWindow &&
1.600 + type==iPrevClickEventType &&
1.601 + (Abs(pos.iX-iPrevClickPos.iX)+Abs(pos.iY-iPrevClickPos.iY))<iDoubleClickMaxDistance &&
1.602 + aEvent.Time()<(iPrevClickTime+iDoubleClickMaxInterval))
1.603 + {
1.604 + aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
1.605 + iPrevClickWindow=NULL; // Set to NULL to block a double double click
1.606 + }
1.607 + else
1.608 + {
1.609 + iPrevClickWindow=iCurrentWindow;
1.610 + }
1.611 + iPrevClickEventType=type;
1.612 + iPrevClickPos=pos;
1.613 + iPrevClickTime=aEvent.Time();
1.614 + }
1.615 + break;
1.616 + default:
1.617 + break;
1.618 + }
1.619 + if (iRepeatWindow)
1.620 + {
1.621 + if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
1.622 + {
1.623 + return;
1.624 + }
1.625 + CancelPointerRepeatEventRequest();
1.626 + }
1.627 + if (QueuePointerEvent(iCurrentWindow, aEvent))
1.628 + {
1.629 + return;
1.630 + }
1.631 + if (iCurrentWindow->DragDropCapture())
1.632 + {
1.633 + aEvent.SetType(EEventDragDrop);
1.634 + QueuePointerEvent(iActualWinPointerIsOver, aEvent);
1.635 + }
1.636 + }
1.637 + }
1.638 +
1.639 +TInt PointerTimerCallBack(TAny *)
1.640 + {
1.641 + TWsPointer::TimerExpired();
1.642 + return(KErrNone);
1.643 + }
1.644 +
1.645 +void TWsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
1.646 + {
1.647 + CScreen* screen = iRootWindow->Screen();
1.648 + WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
1.649 +#if defined(__WINS__)
1.650 + if (aWithinDrawableArea)
1.651 + {
1.652 + if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
1.653 + {
1.654 + return; //Not in the drawable area so user may be trying to click on facia button.
1.655 + }
1.656 + }
1.657 +#endif
1.658 + TRect validRect=screen->GetPointerCursorArea(screen->ScreenSizeMode());
1.659 + if (aPos.iX<validRect.iTl.iX)
1.660 + aPos.iX=validRect.iTl.iX;
1.661 + else if (aPos.iX>=validRect.iBr.iX)
1.662 + aPos.iX=validRect.iBr.iX-1;
1.663 + if (aPos.iY<validRect.iTl.iY)
1.664 + aPos.iY=validRect.iTl.iY;
1.665 + else if (aPos.iY>=validRect.iBr.iY)
1.666 + aPos.iY=validRect.iBr.iY-1;
1.667 + }
1.668 +
1.669 +#if defined(__WINS__)
1.670 +TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
1.671 +#else
1.672 +TBool TWsPointer::PreProcessDriverEvent(TRawEvent &aRawEvent)
1.673 +#endif
1.674 + {
1.675 + TRawEvent::TType type=aRawEvent.Type();
1.676 +
1.677 + if (!IsPointerEventType(type))
1.678 + return ETrue;
1.679 +
1.680 + if (!XyInput())
1.681 + {
1.682 + return EFalse;
1.683 + }
1.684 +
1.685 + //** Log the type, pointer number, and its coordinates
1.686 +#ifdef LOG_WSERV_EVENTS
1.687 + RDebug::Printf("_WSEVENT_POINTER: Pointer number = %d RawEvent Type = %d Coordinates [%d, %d]",
1.688 + aRawEvent.PointerNumber(), type, aRawEvent.Pos().iX, aRawEvent.Pos().iY);
1.689 +#endif
1.690 +
1.691 + // check correctness of aRawEvent.PointerNumber()
1.692 + if (iMaxPointers > 1)
1.693 + {
1.694 + if (aRawEvent.PointerNumber() >= iMaxPointers)
1.695 + {
1.696 + return EFalse;
1.697 + }
1.698 + }
1.699 + else
1.700 + {
1.701 + aRawEvent.SetPointerNumber(0);
1.702 + }
1.703 +
1.704 + if (type != TRawEvent::EPointer3DOutOfRange)
1.705 + {
1.706 + // operations on coordinates are valid for all types except EPointer3DOutOfRange
1.707 + TPoint xy=aRawEvent.Pos();
1.708 + if (DeltaMouse())
1.709 + {
1.710 + #if defined(__WINS__)
1.711 + if (aFromHardware)
1.712 + return EFalse;
1.713 + #endif
1.714 + if (type==TRawEvent::EPointerMove)
1.715 + {
1.716 + xy+=iPointers[aRawEvent.PointerNumber()].iPos;
1.717 + ShiftYCoordinate(xy.iY);
1.718 + RestrictPos(xy);
1.719 + }
1.720 + else
1.721 + xy=iPointers[aRawEvent.PointerNumber()].iPos;
1.722 + }
1.723 + else
1.724 + {
1.725 + #if !defined(__WINS__)
1.726 + TranslateCoordsOnRotation(xy);
1.727 + #else
1.728 + if(iEmulatorRotatePointerCoords)
1.729 + {
1.730 + //emulators that support rotated drawing and touch may want to enable
1.731 + //rotation of pointer events in wsini.
1.732 + TranslateCoordsOnRotation(xy);
1.733 + }
1.734 + #endif
1.735 + CScreen* screen=iRootWindow->Screen();
1.736 + ShiftYCoordinate(xy.iY);
1.737 + // Move the raw event position by shifting it by Origin and scale
1.738 + xy=screen->PhysicalToLogical(xy);
1.739 + RestrictPos(xy);
1.740 + }
1.741 + aRawEvent.Set(type, xy.iX, xy.iY,
1.742 + iIs3DPointer ? aRawEvent.Pos3D().iZ : 0);
1.743 + }
1.744 +
1.745 +#ifdef LOG_WSERV_EVENTS
1.746 + RDebug::Printf("_WSEVENT_POINTER: Coordinates after Rotation and shift [%d, %d]",
1.747 + aRawEvent.Pos().iX, aRawEvent.Pos().iY);
1.748 +#endif
1.749 + return ETrue;
1.750 + }
1.751 +
1.752 +void TWsPointer::TranslateCoordsOnRotation(TPoint& aPoint)
1.753 + {
1.754 + CScreen* screen=iRootWindow->Screen();
1.755 + TSize screenSize=screen->SizeInPixels()-TSize(1,1); //This is in the current rotation
1.756 + switch (screen->Orientation())
1.757 + {
1.758 + case CFbsBitGc::EGraphicsOrientationRotated90:
1.759 + aPoint.SetXY(aPoint.iY,screenSize.iHeight-aPoint.iX);
1.760 + break;
1.761 + case CFbsBitGc::EGraphicsOrientationRotated180:
1.762 + aPoint=-(aPoint-screenSize);
1.763 + break;
1.764 + case CFbsBitGc::EGraphicsOrientationRotated270:
1.765 + aPoint.SetXY(screenSize.iWidth-aPoint.iY,aPoint.iX);
1.766 + break;
1.767 + default:; //To stop warning
1.768 + }
1.769 + }
1.770 +
1.771 +/**
1.772 + * Validates events sent to the Window Server by its Client (Anim or Window Group).
1.773 + * May overwrite aRawEvent's Z coordinate and/or pointer number if Client or digitizer driver
1.774 + * doesn't support them.
1.775 + * @param aRawEvent event to validate
1.776 + * @param aAdvancedPointersEnabled ETrue if Client supports advanced pointer's data (Z coordiante
1.777 + * and pointer number); EFalse otherwise.
1.778 + * @return ETrue if aRawEvent should be processed by Window Server. EFalse if it should be ignored.
1.779 + */
1.780 +TBool TWsPointer::PreProcessClientEvent(TRawEvent &aRawEvent, TBool aAdvancedPointersEnabled)
1.781 + {
1.782 + TRawEvent::TType type=aRawEvent.Type();
1.783 +
1.784 + if (!IsPointerEventType(type))
1.785 + return ETrue;
1.786 +
1.787 + // validate pointer number
1.788 + if (aAdvancedPointersEnabled)
1.789 + {
1.790 + // ignore event if both digitizer driver and Client support pointer numbers, but number is invalid
1.791 + if (aRawEvent.PointerNumber() >= iMaxPointers)
1.792 + {
1.793 + return EFalse;
1.794 + }
1.795 + }
1.796 + else
1.797 + {
1.798 + // set to iPrimaryPointer if Client doesn't support pointer numbers
1.799 + aRawEvent.SetPointerNumber(iPrimaryPointer);
1.800 + }
1.801 +
1.802 + // validate Z coordinate for all pointer events except EPointer3DOutOfRange
1.803 + if (type != TRawEvent::EPointer3DOutOfRange)
1.804 + {
1.805 + if (!iIs3DPointer)
1.806 + {
1.807 + // set Z to 0 for all events when the digitizer driver doesn't support Z coordinate
1.808 + TPoint3D xyz=aRawEvent.Pos3D();
1.809 + aRawEvent.Set(type, xyz.iX, xyz.iY, 0);
1.810 + }
1.811 + else if (!aAdvancedPointersEnabled)
1.812 + {
1.813 + // set Z to actual value if the digitizer driver does support Z coordinate but Client doesn't
1.814 + TPoint3D xyz=aRawEvent.Pos3D();
1.815 + aRawEvent.Set(type, xyz.iX, xyz.iY, iPointers[aRawEvent.PointerNumber()].iPressureProximity);
1.816 + }
1.817 + }
1.818 +
1.819 + return ETrue;
1.820 + }
1.821 +
1.822 +TBool TWsPointer::IsPointerEventType(TRawEvent::TType aType)
1.823 + {
1.824 +#if defined(__WINS__)
1.825 + WS_ASSERT_DEBUG(TRawEvent::EPointerMove==1, EWsPanicRawEventsTypeChanged);
1.826 + WS_ASSERT_DEBUG(TRawEvent::EPointerMove+1==TRawEvent::EPointerSwitchOn, EWsPanicRawEventsTypeChanged);
1.827 + WS_ASSERT_DEBUG(TRawEvent::EPointerSwitchOn+8==TRawEvent::EButton1Down, EWsPanicRawEventsTypeChanged);
1.828 + WS_ASSERT_DEBUG(TRawEvent::EButton1Down+5==TRawEvent::EButton3Up, EWsPanicRawEventsTypeChanged);
1.829 + WS_ASSERT_DEBUG(TRawEvent::EButton3Up+6==TRawEvent::EPointer3DOutOfRange, EWsPanicRawEventsTypeChanged);
1.830 +#endif
1.831 + return (aType == TRawEvent::EPointerMove) ||
1.832 + (aType == TRawEvent::EPointerSwitchOn) ||
1.833 + (aType >= TRawEvent::EButton1Down && aType <= TRawEvent::EButton3Up) ||
1.834 + (aType == TRawEvent::EPointer3DOutOfRange);
1.835 + }
1.836 +
1.837 +/*
1.838 +Pointer Event processing.
1.839 +
1.840 +This method redirects pointer event processing to proper TWsPointer object.
1.841 +*/
1.842 +void TWsPointer::ProcessWsEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
1.843 + {
1.844 + if(iPrimaryPointer!=iPreviousPrimaryPointer)
1.845 + {
1.846 + // The primary pointer may be updated while the TRawEvent is being processed, but this TRawEvent may be
1.847 + // then be consumed by an anim.
1.848 + // If it hasn't then we can leave any repeat request.
1.849 + CancelPointerRepeatEventRequest(iPreviousPrimaryPointer);
1.850 + }
1.851 +
1.852 + if (aEvent.Pointer()->iType == TPointerEvent::EOutOfRange)
1.853 + {
1.854 + iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessOutOfRangeEvent(aEvent, aForceInGroup, aNatural);
1.855 + }
1.856 + else
1.857 + {
1.858 + iPointers[TAdvancedPointerEventHelper::PointerNumber(aEvent)].ProcessEvent(aEvent, aForceInGroup, aNatural);
1.859 + }
1.860 + }
1.861 +
1.862 +/*
1.863 +Pointer Event Processing - stage 1 of 3:
1.864 +- updating this pointer's state: coordinates and grabbing window
1.865 +- updating coordinates of event
1.866 +- updating event type to Drag
1.867 +- simulated moves
1.868 +- generating events: EEnterCloseProximity, EExitCloseProximity, EEnterHighPressure, EExitHighPressure
1.869 +*/
1.870 +void TWsPointer::ProcessEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup,TBool aNatural)
1.871 + {
1.872 + TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();
1.873 + TPointerEvent::TType eventType=pointerEvent.iType;
1.874 +
1.875 + if (iState == EPointerStateOutOfRange)
1.876 + {
1.877 + // new pointer comes in range, so clear information after previous one
1.878 + Clear();
1.879 + }
1.880 +
1.881 + // update coordinates
1.882 + iPos=pointerEvent.iPosition;
1.883 + iPressureProximity=TAdvancedPointerEventHelper::Z(aEvent);
1.884 +
1.885 + if (eventType == TPointerEvent::EMove && !MovesAvailable() && iState != EPointerStateDown)
1.886 + {
1.887 + return;
1.888 + }
1.889 +
1.890 + // re-log and update parent position
1.891 + TPoint parPos;
1.892 + ReLogCurrentWindow(pointerEvent.iPosition,parPos,aForceInGroup);
1.893 + pointerEvent.iParentPosition=parPos;
1.894 +
1.895 +
1.896 +#ifdef LOG_WSERV_EVENTS
1.897 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::ProcessEvent Event to be sent to this window %U", reinterpret_cast<TUint32>(iCurrentWindow));
1.898 + RDebug::Print(_L("_WSEVENT_POINTER: TWsPointer::ProcessEvent EventName %S and Event State %d "), &WsEventName(aEvent), iState);
1.899 +#endif
1.900 + // update state
1.901 + switch(eventType)
1.902 + {
1.903 + case TPointerEvent::EButton1Down:
1.904 + if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
1.905 + {
1.906 + iGrabWindow=iCurrentWindow;
1.907 + }
1.908 + if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
1.909 + {
1.910 + pointerEvent.iType=TPointerEvent::EMove;
1.911 + ProcessEvent(aEvent, EFalse);
1.912 + pointerEvent.iType=TPointerEvent::EButton1Down;
1.913 + }
1.914 + switch(iState)
1.915 + {
1.916 + case EPointerStateOutOfRange:
1.917 + iState = EPointerStateDown;
1.918 + SendEnterExitEvent(EEventPointerEnter);
1.919 + // intentional lack of break statement
1.920 + case EPointerStateUp:
1.921 + iState = EPointerStateDown;
1.922 + ProcessEvent(aEvent, aNatural);
1.923 + if (iPressureProximity >= iEnterHighPressureThreshold)
1.924 + {
1.925 + iInHighPressure = ETrue;
1.926 + pointerEvent.iType = TPointerEvent::EEnterHighPressure;
1.927 + ProcessEvent(aEvent, EFalse);
1.928 + }
1.929 + else
1.930 + {
1.931 + iInHighPressure = EFalse;
1.932 + }
1.933 + break;
1.934 + case EPointerStateDown:
1.935 + if (iInHighPressure && iPressureProximity < iExitHighPressureThreshold)
1.936 + {
1.937 + iInHighPressure = EFalse;
1.938 + eventType = TPointerEvent::EExitHighPressure;
1.939 + }
1.940 + else if (!iInHighPressure && iPressureProximity >= iEnterHighPressureThreshold)
1.941 + {
1.942 + iInHighPressure = ETrue;
1.943 + eventType = TPointerEvent::EEnterHighPressure;
1.944 + }
1.945 + ProcessEvent(aEvent, aNatural);
1.946 + break;
1.947 + }
1.948 + break;
1.949 + case TPointerEvent::EButton1Up:
1.950 + iGrabWindow=NULL;
1.951 + switch(iState)
1.952 + {
1.953 + case EPointerStateDown:
1.954 + iState = EPointerStateUp;
1.955 + ProcessEvent(aEvent, aNatural);
1.956 + if (iPressureProximity < iExitCloseProximityThreshold)
1.957 + {
1.958 + pointerEvent.iType = TPointerEvent::EExitCloseProximity;
1.959 + iInCloseProximity = EFalse;
1.960 + ProcessEvent(aEvent, EFalse);
1.961 + }
1.962 + else
1.963 + {
1.964 + iInCloseProximity = ETrue;
1.965 + }
1.966 + break;
1.967 + case EPointerStateOutOfRange:
1.968 + iState = EPointerStateUp;
1.969 + SendEnterExitEvent(EEventPointerEnter);
1.970 + // intentional lack of break statement
1.971 + case EPointerStateUp:
1.972 + if (iInCloseProximity &&
1.973 + iPressureProximity < iExitCloseProximityThreshold)
1.974 + {
1.975 + iInCloseProximity = EFalse;
1.976 + pointerEvent.iType = TPointerEvent::EExitCloseProximity;
1.977 + }
1.978 + else if (!iInCloseProximity &&
1.979 + iPressureProximity >= iEnterCloseProximityThreshold)
1.980 + {
1.981 + iInCloseProximity = ETrue;
1.982 + pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
1.983 + }
1.984 + ProcessEvent(aEvent, aNatural);
1.985 + break;
1.986 + }
1.987 + break;
1.988 + case TPointerEvent::EMove:
1.989 + switch(iState)
1.990 + {
1.991 + case EPointerStateDown:
1.992 + if (iInHighPressure &&
1.993 + iPressureProximity < iExitHighPressureThreshold)
1.994 + {
1.995 + iInHighPressure = EFalse;
1.996 + pointerEvent.iType = TPointerEvent::EExitHighPressure;
1.997 + }
1.998 + else if (!iInHighPressure &&
1.999 + iPressureProximity >= iEnterHighPressureThreshold)
1.1000 + {
1.1001 + iInHighPressure = ETrue;
1.1002 + pointerEvent.iType = TPointerEvent::EEnterHighPressure;
1.1003 + }
1.1004 + else
1.1005 + {
1.1006 + pointerEvent.iType = TPointerEvent::EDrag;
1.1007 + }
1.1008 + break;
1.1009 + case EPointerStateUp:
1.1010 + if (iInCloseProximity &&
1.1011 + iPressureProximity < iExitCloseProximityThreshold)
1.1012 + {
1.1013 + iInCloseProximity = EFalse;
1.1014 + pointerEvent.iType = TPointerEvent::EExitCloseProximity;
1.1015 + }
1.1016 + else if (!iInCloseProximity &&
1.1017 + iPressureProximity >= iEnterCloseProximityThreshold)
1.1018 + {
1.1019 + iInCloseProximity = ETrue;
1.1020 + pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
1.1021 + }
1.1022 + break;
1.1023 + case EPointerStateOutOfRange:
1.1024 + iState = EPointerStateUp;
1.1025 + SendEnterExitEvent(EEventPointerEnter);
1.1026 + if (iPressureProximity >= iEnterCloseProximityThreshold)
1.1027 + {
1.1028 + iInCloseProximity = ETrue;
1.1029 + pointerEvent.iType = TPointerEvent::EEnterCloseProximity;
1.1030 + }
1.1031 + break;
1.1032 + }
1.1033 + ProcessEvent(aEvent, aNatural);
1.1034 + break;
1.1035 + default:
1.1036 + ProcessEvent(aEvent, aNatural);
1.1037 + break;
1.1038 + }
1.1039 + }
1.1040 +
1.1041 +/**
1.1042 +Processes OutOfRange events:
1.1043 +- injects event to key click plugin
1.1044 +- directs event to the last current window
1.1045 +- sends Exit event
1.1046 +*/
1.1047 +void TWsPointer::ProcessOutOfRangeEvent(TWsEvent& aEvent,const CWsWindowGroup* aForceInGroup, TBool aNatural)
1.1048 + {
1.1049 + if (iState != EPointerStateOutOfRange)
1.1050 + {
1.1051 +#ifdef LOG_WSERV_EVENTS
1.1052 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::ProcessOutOfRangeEvent Pointer Number = %d, iState =%d ", iNumber, iState);
1.1053 +#endif
1.1054 +
1.1055 + // OutOfRange event generated by driver doesn't contain correct coordinates,
1.1056 + // we update them from last state in order to deliver event to the proper window.
1.1057 + SendEnterExitEvent(EEventPointerExit);
1.1058 +
1.1059 + iState = EPointerStateOutOfRange;
1.1060 +
1.1061 + TAdvancedPointerEventHelper::SetPointerNumber(aEvent, iNumber);
1.1062 + TAdvancedPointerEventHelper::SetZ(aEvent, iPressureProximity);
1.1063 +
1.1064 + TAdvancedPointerEvent& pointerEvent = *aEvent.Pointer();
1.1065 + iCurrentWindow=iRootWindow->PointerWindow(iPos,&pointerEvent.iPosition,&pointerEvent.iParentPosition,iGrabWindow,
1.1066 + iActualWinPointerIsOver,aForceInGroup);
1.1067 + ProcessEvent(aEvent, aNatural);
1.1068 + }
1.1069 + }
1.1070 +
1.1071 +void TWsPointer::NotifyCClick(TAdvancedPointerEvent& aPointerEvent)
1.1072 + {
1.1073 + if (CClick::IsHandler())
1.1074 + {
1.1075 + CClick::PointerEvent(iPos,aPointerEvent);
1.1076 + TPointerEventData params;
1.1077 + params.iVersion=0;
1.1078 + params.iCurrentPos=iPos;
1.1079 + TAdvancedPointerEventHelper::Copy(aPointerEvent, params.iPointerEvent);
1.1080 + params.iClientHandle=iCurrentWindow->ClientHandle();
1.1081 + params.iWindowOrigin=iCurrentWindow->Origin();
1.1082 + CWsWindowGroup* groupWin=iCurrentWindow->WinGroup();
1.1083 + params.iWindowGroupId=groupWin ? groupWin->Identifier() : 0;
1.1084 + params.iSource=TPointerEventData::EUnspecified;
1.1085 + CClick::OtherEvent(EEventPointer,¶ms);
1.1086 + }
1.1087 + }
1.1088 +
1.1089 +/*
1.1090 +Pointer Event Processing - stage 2 of 3:
1.1091 +- injecting events to key click plugin
1.1092 +- pointer filtering
1.1093 +- injection to event buffer
1.1094 +- injection to keyboard emulator
1.1095 +- clearing iCurrentWindow if going up and !MovesAvailable()
1.1096 +- updating pointer cursor
1.1097 +*/
1.1098 +void TWsPointer::ProcessEvent(TWsEvent& aEvent, TBool aNatural)
1.1099 + {
1.1100 + if (aNatural)
1.1101 + {
1.1102 + NotifyCClick(*aEvent.Pointer());
1.1103 + }
1.1104 +
1.1105 + TUint filter=iCurrentWindow->PointerFilter();
1.1106 + TPointerEvent::TType type=aEvent.Pointer()->iType;
1.1107 + if ((type!=TPointerEvent::EMove || !(filter&EPointerFilterMove)) &&
1.1108 + (type!=TPointerEvent::EDrag || !(filter&EPointerFilterDrag)))
1.1109 + {
1.1110 + if (iNumber == iPrimaryPointer)
1.1111 + {
1.1112 + TPoint pos=aEvent.Pointer()->iPosition;
1.1113 + if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
1.1114 + {
1.1115 + CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
1.1116 + }
1.1117 + else if ((type==TPointerEvent::EEnterCloseProximity || type==TPointerEvent::EExitCloseProximity ||
1.1118 + type==TPointerEvent::EEnterHighPressure || type==TPointerEvent::EExitHighPressure) &&
1.1119 + iCurrentWindow->UsingPointerBuffer())
1.1120 + {
1.1121 + CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
1.1122 + ProcessPointerEvent(aEvent);
1.1123 + }
1.1124 + else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
1.1125 + {
1.1126 +#ifdef LOG_WSERV_EVENTS
1.1127 + RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for primary pointer");
1.1128 +#endif
1.1129 + ProcessPointerEvent(aEvent);
1.1130 + }
1.1131 + }
1.1132 + else if (!iCurrentWindow->UsingPointerBuffer() || (type != TPointerEvent::EMove && type != TPointerEvent::EDrag))
1.1133 + {
1.1134 +#ifdef LOG_WSERV_EVENTS
1.1135 + RDebug::Printf("_WSEVENT_POINTER: Calling ProcessPointerEvent for non primary pointer");
1.1136 +#endif
1.1137 + ProcessPointerEvent(aEvent);
1.1138 + }
1.1139 + }
1.1140 + if (!MovesAvailable() && (type==TPointerEvent::EButton1Up ||
1.1141 + type==TPointerEvent::ESwitchOn ||
1.1142 + type==TPointerEvent::EOutOfRange))
1.1143 + {
1.1144 + iCurrentWindow=NULL;
1.1145 + }
1.1146 + if (iNumber == iPrimaryPointer)
1.1147 + {
1.1148 + PointerCursorUpdateCheck();
1.1149 + }
1.1150 + }
1.1151 +
1.1152 +void TWsPointer::TimerExpired()
1.1153 + {
1.1154 + WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
1.1155 + if (iUpdateRequired)
1.1156 + {
1.1157 + UpdatePointerCursor();
1.1158 + iUpdateRequired=EFalse;
1.1159 + }
1.1160 + else
1.1161 + {
1.1162 + iTimerQueued=EFalse;
1.1163 + iPeriodicTimer->Cancel();
1.1164 + }
1.1165 + }
1.1166 +
1.1167 +void TWsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
1.1168 + {
1.1169 + aTime=iDoubleClickMaxInterval;
1.1170 + aDistance=iDoubleClickMaxDistance;
1.1171 + }
1.1172 +
1.1173 +void TWsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
1.1174 + {
1.1175 + iDoubleClickMaxInterval=aTime;
1.1176 + iDoubleClickMaxDistance=aDistance;
1.1177 + }
1.1178 +
1.1179 +void TWsPointer::PointerCursorUpdateCheck()
1.1180 + {
1.1181 + CWsPointerCursor* sprite=CalculatePointerCursor();
1.1182 + if (iCursorSprite || sprite) // If there either was, or is a pointer cursor we need an update
1.1183 + {
1.1184 + if (!iTimerQueued)
1.1185 + {
1.1186 + UpdatePointerCursorTo(sprite);
1.1187 + iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1.1188 + TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1.1189 + TCallBack(PointerTimerCallBack,NULL));
1.1190 + iTimerQueued=ETrue;
1.1191 + }
1.1192 + else
1.1193 + {
1.1194 + iUpdateRequired=ETrue;
1.1195 + }
1.1196 + }
1.1197 + }
1.1198 +
1.1199 +void TWsPointer::UpdatePointerCursor()
1.1200 + {
1.1201 +//__PROFILE_START(3);
1.1202 + CWsPointerCursor* sprite=iPointers[iPrimaryPointer].CalculatePointerCursor();
1.1203 + UpdatePointerCursorTo(sprite);
1.1204 +//__PROFILE_END(3);
1.1205 + }
1.1206 +
1.1207 +void TWsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
1.1208 + {
1.1209 + if (iCursorSprite!=aNewCursor)
1.1210 + {
1.1211 + if (iCursorSprite)
1.1212 + {
1.1213 + iCursorSprite->Deactivate();
1.1214 + }
1.1215 + iCursorSprite=aNewCursor;
1.1216 + if (iCursorSprite)
1.1217 + {
1.1218 + iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
1.1219 + iCursorSprite->Activate();
1.1220 + }
1.1221 + }
1.1222 + else if (iCursorSprite)
1.1223 + {
1.1224 + iCursorSprite->SetPos(iPointers[iPrimaryPointer].iPos);
1.1225 + }
1.1226 + }
1.1227 +
1.1228 +CWsPointerCursor* TWsPointer::CalculatePointerCursor()
1.1229 + {
1.1230 + CWsPointerCursor* sprite=NULL;
1.1231 + if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
1.1232 + {
1.1233 + const CWsWindowBase* window=iCurrentWindow;
1.1234 + do {
1.1235 + sprite=window->PointerCursor();
1.1236 + if (window->WinType()!=EWinTypeClient)
1.1237 + {
1.1238 + break;
1.1239 + }
1.1240 + window=window->BaseParent();
1.1241 + } while (!sprite);
1.1242 + }
1.1243 + if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
1.1244 + {
1.1245 + sprite=CWsClient::DefaultSystemPointerCursor();
1.1246 + }
1.1247 + return sprite;
1.1248 + }
1.1249 +
1.1250 +/*
1.1251 +Callback function for event queue walk
1.1252 +*/
1.1253 +TEventQueueWalkRet PointerRepeatPurgeFunc(TAny* aReqPtrNum, TWsEvent* aQueueEvent)
1.1254 + {
1.1255 + return(TWsPointer::PointerRepeatPurgeCheck(aQueueEvent, reinterpret_cast<TUint>(aReqPtrNum)));
1.1256 + }
1.1257 +
1.1258 +TBool TWsPointer::PointerEventRepeatCheck(const TWsEvent* aEvent, TUint32 aHandle)
1.1259 +//
1.1260 +// Return ETrue if this pointer event is consumed by the pointer repeat
1.1261 +//
1.1262 + {
1.1263 + // Must be a pointer event type in order to get the pointer number,
1.1264 + // which is needed to check the repeat window.
1.1265 + WS_ASSERT_DEBUG(aEvent->Type()==EEventPointer,EWsPanicEventType);
1.1266 +
1.1267 + const TAdvancedPointerEvent* pntEvent=aEvent->Pointer();
1.1268 + if ( TAdvancedPointerEventHelper::PointerNumber(*aEvent)==iNumber &&
1.1269 + aHandle==iRepeatWindow->ClientHandle())
1.1270 + {
1.1271 + switch(pntEvent->iType)
1.1272 + {
1.1273 + case TPointerEvent::EDrag: // deliberate drop-through
1.1274 + case TPointerEvent::EMove: // deliberate drop-through
1.1275 + case TPointerEvent::EEnterCloseProximity: // deliberate drop-through
1.1276 + case TPointerEvent::EExitCloseProximity: // deliberate drop-through
1.1277 + case TPointerEvent::EEnterHighPressure: // deliberate drop-through
1.1278 + case TPointerEvent::EExitHighPressure:
1.1279 + {
1.1280 + if(iRepeatRect.Contains(pntEvent->iPosition))
1.1281 + {
1.1282 + return(ETrue);
1.1283 + }
1.1284 + break;
1.1285 + }
1.1286 + default:
1.1287 + // do nothing and drop through
1.1288 + break;
1.1289 + }
1.1290 + }
1.1291 + return(EFalse);
1.1292 + }
1.1293 +
1.1294 +TEventQueueWalkRet TWsPointer::PointerRepeatPurgeCheck(TWsEvent* aQueueEvent, TUint8 aReqPtrNum)
1.1295 + {
1.1296 + // Return value is "WalkOK", unless a repeated event is found that needs to be deleted.
1.1297 + TEventQueueWalkRet eventQueueWalkRet(EEventQueueWalkOk);
1.1298 +
1.1299 + // Check the WSEvent Type
1.1300 + if (aQueueEvent->Type()==EEventPointer) // aEvent is a pointer event
1.1301 + {
1.1302 + // It is a pointer event, so we can get the pointer number
1.1303 + // to check if there is a repeat request for that pointer.
1.1304 + const TInt eventPtrNum(TAdvancedPointerEventHelper::PointerNumber(*aQueueEvent));
1.1305 + // If aEvent's pointer has an active repeat request,
1.1306 + // then it'll have a repeat window.
1.1307 + if ((eventPtrNum == aReqPtrNum) && RepeatWindow(eventPtrNum))
1.1308 + {
1.1309 + // There is a repeat request for the pointer.
1.1310 + // Is there a queued repeated event to be deleted?
1.1311 + TWsPointer& wsPointer = iPointers[eventPtrNum];
1.1312 + if (wsPointer.PointerEventRepeatCheck(aQueueEvent,aQueueEvent->Handle()))
1.1313 + {
1.1314 + // Update the return value to purge the event
1.1315 + // as it is a move/drag within the repeat rect
1.1316 + eventQueueWalkRet=EEventQueueWalkDeleteEvent;
1.1317 + }
1.1318 + else
1.1319 + {
1.1320 + // No queued repeated event was found, therefore the
1.1321 + // request is still pending and needs to be cancelled.
1.1322 + wsPointer.CancelPointerRepeatEventRequest();
1.1323 + }
1.1324 + }
1.1325 + }
1.1326 + return eventQueueWalkRet;
1.1327 + }
1.1328 +
1.1329 +void TWsPointer::RequestRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
1.1330 + {
1.1331 + CancelPointerRepeatEventRequest();
1.1332 + iRepeatWindow=aWindow;
1.1333 + iRepeatRect=aRequest.rect;
1.1334 + iRepeatTimer->After(aRequest.time);
1.1335 + aWindow->EventQueue()->WalkEventQueue(&PointerRepeatPurgeFunc,reinterpret_cast<TInt*>(iNumber));
1.1336 + if (iRepeatWindow && !iRepeatRect.Contains(iPos-iRepeatWindow->Origin()))
1.1337 + {
1.1338 + CancelPointerRepeatEventRequest();
1.1339 + }
1.1340 + }
1.1341 +
1.1342 +TInt TWsPointer::RequestPointerRepeatEvent(CWsWindow* aWindow, const TWsWinCmdRequestPointerRepeatEvent& aRequest)
1.1343 + {
1.1344 + TInt errNo = KErrNone;
1.1345 + TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
1.1346 + if(PointerNumberInRange(pointerNum))
1.1347 + {
1.1348 + iPointers[pointerNum].RequestRepeatEvent(aWindow,aRequest);
1.1349 + }
1.1350 + else
1.1351 + {
1.1352 + errNo=KErrArgument;
1.1353 + }
1.1354 + return errNo;
1.1355 + }
1.1356 +
1.1357 +void TWsPointer::CancelPointerRepeatEventRequest()
1.1358 + {
1.1359 + if (iRepeatWindow)
1.1360 + {
1.1361 + iRepeatWindow=NULL;
1.1362 + if(iRepeatTimer)
1.1363 + iRepeatTimer->Cancel();
1.1364 + }
1.1365 + }
1.1366 +
1.1367 +TInt TWsPointer::CancelPointerRepeatEventRequest(const TWsWinCmdCancelPointerRepeatEventRequest& aRequest)
1.1368 + {
1.1369 + TInt errNo = KErrNone;
1.1370 + TUint8 pointerNum = aRequest.HasPointerNumber() ? aRequest.pointerNumber : iPrimaryPointer;
1.1371 + if(PointerNumberInRange(pointerNum))
1.1372 + {
1.1373 + iPointers[pointerNum].CancelPointerRepeatEventRequest();
1.1374 + }
1.1375 + else
1.1376 + {
1.1377 + errNo=KErrArgument;
1.1378 + }
1.1379 + return errNo;
1.1380 + }
1.1381 +
1.1382 +void TWsPointer::RepeatTimerCompleted()
1.1383 + {
1.1384 + TWsEvent event;
1.1385 + event.SetType(EEventPointer);
1.1386 + event.SetTimeNow();
1.1387 + TPoint3D point3D(iPos-iRepeatWindow->Origin());
1.1388 + point3D.iZ=iPressureProximity;
1.1389 + TAdvancedPointerEventHelper::InitAdvancedPointerEvent(event,
1.1390 + TPointerEvent::EButtonRepeat,
1.1391 + TWindowServerEvent::GetModifierState(),
1.1392 + point3D,
1.1393 + (iPos-iRepeatWindow->BaseParent()->Origin()),
1.1394 + iNumber);
1.1395 + QueuePointerEvent(iRepeatWindow, event);
1.1396 + iRepeatWindow=NULL;
1.1397 + }
1.1398 +
1.1399 +#if defined(__WINS__)
1.1400 +void TWsPointer::SetXyInputType(TXYInputType aXyInputType)
1.1401 + {
1.1402 + if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse)
1.1403 + {
1.1404 + // change from Mouse types to Pointer/None
1.1405 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.1406 + {
1.1407 + if (iPointers[ii].iState != EPointerStateDown)
1.1408 + {
1.1409 + iPointers[ii].iCurrentWindow=NULL;
1.1410 + }
1.1411 + }
1.1412 + UpdatePointerCursor();
1.1413 + }
1.1414 + else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer)
1.1415 + {
1.1416 + // change from Pointer/None types to Mouse types
1.1417 + for (TInt ii = 0; ii < iMaxPointers; ii++)
1.1418 + {
1.1419 + if (iPointers[ii].iState != EPointerStateDown)
1.1420 + {
1.1421 + TPoint pos(iPointers[ii].iPos);
1.1422 + TPoint parPos;
1.1423 + iPointers[ii].ReLogCurrentWindow(pos,parPos,NULL);
1.1424 + }
1.1425 + }
1.1426 + UpdatePointerCursor();
1.1427 + }
1.1428 + iXyInputType=aXyInputType;
1.1429 + }
1.1430 +#endif
1.1431 +
1.1432 +/**
1.1433 +Updates Primary Pointer before aRawEvent is processed. Only events related to Primary
1.1434 +Pointer will be sent to Clients which require single pointer environment.
1.1435 +
1.1436 +This method implements single pointer environment emulation rules (see design
1.1437 +documentation for more details).
1.1438 +
1.1439 +@param aRawEvent Incoming event used to update the Primary Pointer. It must be a pointer event,
1.1440 + as defined by TWsPointer::IsPointerEventType(TRawEvent::TType).
1.1441 +*/
1.1442 +void TWsPointer::UpdatePrimaryPointer(const TRawEvent& aRawEvent)
1.1443 + {
1.1444 + iPreviousPrimaryPointer = iPrimaryPointer;
1.1445 +
1.1446 + TRawEvent::TType type=aRawEvent.Type();
1.1447 + TInt pointerNumber = aRawEvent.PointerNumber();
1.1448 +
1.1449 +#ifdef LOG_WSERV_EVENTS
1.1450 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer Current Primary pointer = %d",iPrimaryPointer);
1.1451 + for(TInt i=0; i< iMaxPointers; i++)
1.1452 + {
1.1453 + 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);
1.1454 + }
1.1455 +#endif
1.1456 +
1.1457 + // If primary pointer is out of range, then the first pointer that will
1.1458 + // start being detected (come back in range) will become primary.
1.1459 + if (iPointers[iPrimaryPointer].iState == EPointerStateOutOfRange)
1.1460 + {
1.1461 + if (type != TRawEvent::EPointer3DOutOfRange && iPointers[pointerNumber].iState == EPointerStateOutOfRange)
1.1462 + {
1.1463 + iPrimaryPointer = pointerNumber;
1.1464 +#ifdef LOG_WSERV_EVENTS
1.1465 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(OutRange) = %d",iPrimaryPointer);
1.1466 +#endif
1.1467 + }
1.1468 + return;
1.1469 + }
1.1470 +
1.1471 + // if non-primary pointer sends EButton1Down event, and actual primary pointer
1.1472 + // is not down, then the pointer which has sent EButton1Down becomes primary.
1.1473 + if (type == TRawEvent::EButton1Down &&
1.1474 + iPointers[iPrimaryPointer].iState != EPointerStateDown)
1.1475 + {
1.1476 + iPrimaryPointer = pointerNumber;
1.1477 +#ifdef LOG_WSERV_EVENTS
1.1478 + RDebug::Printf("_WSEVENT_POINTER: TWsPointer::UpdatePrimaryPointer New Primary pointer(ButtonDown) = %d",iPrimaryPointer);
1.1479 +#endif
1.1480 + return;
1.1481 + }
1.1482 + }
1.1483 +
1.1484 +/** Sets Z coordinate threshold values for TPointerEvent::EEnterCloseProximity
1.1485 +and TPointerEvent::EExitCloseProximity events.
1.1486 +@return KErrNone if successful,
1.1487 + KErrNotSupported if the device doesn't support threshold values,
1.1488 + KErrArgument if aEnterCloseProximityThreshold is less than aExitCloseProximityThreshold
1.1489 +@see RWsSession::SetCloseProximityThresholds which calls this method
1.1490 +*/
1.1491 +TInt TWsPointer::SetCloseProximityThresholds(TInt aEnterCloseProximityThreshold, TInt aExitCloseProximityThreshold)
1.1492 + {
1.1493 + if (aEnterCloseProximityThreshold < aExitCloseProximityThreshold)
1.1494 + {
1.1495 + return KErrArgument;
1.1496 + }
1.1497 +
1.1498 + TInt ret = HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, aEnterCloseProximityThreshold);
1.1499 + if (ret != KErrNone)
1.1500 + {
1.1501 + return ret;
1.1502 + }
1.1503 +
1.1504 + ret = HAL::Set(HALData::EPointer3DExitCloseProximityThreshold, aExitCloseProximityThreshold);
1.1505 + WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicProxThresholdsInconsist);
1.1506 + if (ret != KErrNone)
1.1507 + {
1.1508 + HAL::Set(HALData::EPointer3DEnterCloseProximityThreshold, iEnterCloseProximityThreshold);
1.1509 + return ret;
1.1510 + }
1.1511 +
1.1512 + iEnterCloseProximityThreshold = aEnterCloseProximityThreshold;
1.1513 + iExitCloseProximityThreshold = aExitCloseProximityThreshold;
1.1514 + return KErrNone;
1.1515 + }
1.1516 +
1.1517 +/**
1.1518 +@return Z coordinate threshold value for TPointerEvent::EEnterCloseProximity events
1.1519 +@see RWsSession::GetEnterCloseProximityThreshold which calls this method
1.1520 +*/
1.1521 +TInt TWsPointer::GetEnterCloseProximityThreshold()
1.1522 + {
1.1523 + return iEnterCloseProximityThreshold;
1.1524 + }
1.1525 +
1.1526 +/**
1.1527 +@return Z coordinate threshold value for TPointerEvent::EExitCloseProximity events
1.1528 +@see RWsSession::GetExitCloseProximityThreshold which calls this method
1.1529 +*/
1.1530 +TInt TWsPointer::GetExitCloseProximityThreshold()
1.1531 + {
1.1532 + return iExitCloseProximityThreshold;
1.1533 + }
1.1534 +
1.1535 +/** Sets Z coordinate threshold value for TPointerEvent::EEnterHighPressure and
1.1536 +TPointerEvent::EExitHighPressure events.
1.1537 +@return KErrNone if successful,
1.1538 + KErrNotSupported if the device doesn't support threshold values,
1.1539 + KErrArgument if aEnterHighPressureThreshold is less than aExitHighPressureThreshold
1.1540 +@see RWsSession::SetHighPressureThresholds which calls this method
1.1541 +*/
1.1542 +TInt TWsPointer::SetHighPressureThresholds(TInt aEnterHighPressureThreshold, TInt aExitHighPressureThreshold)
1.1543 + {
1.1544 + if (aEnterHighPressureThreshold < aExitHighPressureThreshold)
1.1545 + {
1.1546 + return KErrArgument;
1.1547 + }
1.1548 +
1.1549 + TInt ret = HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, aEnterHighPressureThreshold);
1.1550 + if (ret != KErrNone)
1.1551 + {
1.1552 + return ret;
1.1553 + }
1.1554 +
1.1555 + ret = HAL::Set(HALData::EPointer3DExitHighPressureThreshold, aExitHighPressureThreshold);
1.1556 + WS_ASSERT_DEBUG(ret == KErrNone, EWsPanicPressThresholdsInconsist);
1.1557 + if (ret != KErrNone)
1.1558 + {
1.1559 + HAL::Set(HALData::EPointer3DEnterHighPressureThreshold, iEnterHighPressureThreshold);
1.1560 + return ret;
1.1561 + }
1.1562 +
1.1563 + iEnterHighPressureThreshold = aEnterHighPressureThreshold;
1.1564 + iExitHighPressureThreshold = aExitHighPressureThreshold;
1.1565 + return KErrNone;
1.1566 + }
1.1567 +
1.1568 +/**
1.1569 +@return Z coordinate threshold value for TPointerEvent::EEnterHighPressure events
1.1570 +@see RWsSession::GetEnterHighPressureThreshold which calls this method
1.1571 +*/
1.1572 +TInt TWsPointer::GetEnterHighPressureThreshold()
1.1573 + {
1.1574 + return iEnterHighPressureThreshold;
1.1575 + }
1.1576 +
1.1577 +/**
1.1578 +@return Z coordinate threshold value for TPointerEvent::EExitHighPressure events
1.1579 +@see RWsSession::GetExitHighPressureThreshold which calls this method
1.1580 +*/
1.1581 +TInt TWsPointer::GetExitHighPressureThreshold()
1.1582 + {
1.1583 + return iExitHighPressureThreshold;
1.1584 + }
1.1585 +
1.1586 +
1.1587 +/**
1.1588 +This function is specific for capacitive touch screens, where user's finger is the pointer device.
1.1589 +Usability studies have shown that the user's perception of the location of the pointer hit is always
1.1590 +away from few pixels north of the actual hit centre as detected by the digitizer device. So, this function
1.1591 +will shift all pointer events by a specified Y displacement.
1.1592 +
1.1593 +@param aY Current y coordinate pointer position.
1.1594 +
1.1595 + */
1.1596 +void TWsPointer::ShiftYCoordinate(TInt& aY)
1.1597 + {
1.1598 + WS_ASSERT_DEBUG(iYOffset>=0, EWsPanicInvalidPointerOffset);
1.1599 + if (aY >= iYOffset)
1.1600 + {
1.1601 + aY -=iYOffset;
1.1602 + }
1.1603 + else
1.1604 + {
1.1605 + aY=0;
1.1606 + }
1.1607 + }
1.1608 +//
1.1609 +CWsPointerTimer::CWsPointerTimer(MPointerTimerCallback& aPointerTimerCallback)
1.1610 +: CTimer(EPointerRepeatPriority), iPointerTimerCallback(aPointerTimerCallback)
1.1611 + {}
1.1612 +
1.1613 +void CWsPointerTimer::ConstructL()
1.1614 + {
1.1615 + CTimer::ConstructL();
1.1616 + CActiveScheduler::Add(this);
1.1617 + }
1.1618 +
1.1619 +CWsPointerTimer* CWsPointerTimer::NewL(MPointerTimerCallback& aWsPointer)
1.1620 + {
1.1621 + CWsPointerTimer* self = new(ELeave) CWsPointerTimer(aWsPointer);
1.1622 + CleanupStack::PushL(self);
1.1623 + self->ConstructL();
1.1624 + CleanupStack::Pop(self);
1.1625 + return self;
1.1626 + }
1.1627 +
1.1628 +void CWsPointerTimer::RunL()
1.1629 + {
1.1630 + User::ResetInactivityTime();
1.1631 + WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
1.1632 + iPointerTimerCallback.RepeatTimerCompleted();
1.1633 + }
1.1634 +//
1.1635 +
1.1636 +CWsPointerBuffer::~CWsPointerBuffer()
1.1637 + {
1.1638 + if (this == iCurrentBuffer)
1.1639 + {
1.1640 + // We're about to be destroyed - don't want to be pointed at any more.
1.1641 + iCurrentBuffer = NULL;
1.1642 + }
1.1643 + iList.Remove(*this);
1.1644 + }
1.1645 +
1.1646 +void CWsPointerBuffer::ConnectL(CWsClientWindow* aWindow, TInt aMaxPoints, TUint aFlags)
1.1647 + {
1.1648 + CWsPointerBuffer* pb=NULL;
1.1649 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.1650 + {
1.1651 + if (pb->iWindow==aWindow)
1.1652 + {
1.1653 + User::Leave(KErrInUse);
1.1654 + }
1.1655 + }
1.1656 + CWsPointerBuffer* pbuf=new(ELeave) CWsPointerBuffer;
1.1657 + pbuf->iWindow=aWindow;
1.1658 + pbuf->iMaxPoints=aMaxPoints;
1.1659 + pbuf->iFlags=aFlags;
1.1660 + iList.AddFirst(*pbuf);
1.1661 + CleanupStack::PushL(pbuf);
1.1662 + AdjustMaxSizeL();
1.1663 + CleanupStack::Pop();
1.1664 + }
1.1665 +
1.1666 +void CWsPointerBuffer::Disconnect(CWsClientWindow* aWindow)
1.1667 + {
1.1668 + CWsPointerBuffer* pb=NULL;
1.1669 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.1670 + {
1.1671 + if (pb->iWindow==aWindow)
1.1672 + {
1.1673 + delete pb; // Note that the destructor also sets iCurrentBuffer to NULL if it is pointing at pb
1.1674 + TRAP_IGNORE(AdjustMaxSizeL()); // Shouldn't fail, but doesn't matter if it does as we simply have a larger buffer than needed
1.1675 + break; // from for loop
1.1676 + }
1.1677 + }
1.1678 + }
1.1679 +
1.1680 +void CWsPointerBuffer::Reset()
1.1681 + {
1.1682 + iSignalled=EFalse;
1.1683 + iPointerBuffer->Reset();
1.1684 + }
1.1685 +
1.1686 +void CWsPointerBuffer::SignalBufferReady()
1.1687 + {
1.1688 + if (!iSignalled)
1.1689 + {
1.1690 + if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
1.1691 + {
1.1692 + iSignalled=ETrue;
1.1693 + }
1.1694 + }
1.1695 + }
1.1696 +
1.1697 +void CWsPointerBuffer::PointerEvent(CWsClientWindow* aWindow,const TPoint &aPoint)
1.1698 + {
1.1699 + if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
1.1700 + {
1.1701 + Reset();
1.1702 + CWsPointerBuffer* pb=NULL;
1.1703 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.1704 + {
1.1705 + if (pb->iWindow==aWindow)
1.1706 + {
1.1707 + iCurrentBuffer=pb;
1.1708 + break; // from for loop
1.1709 + }
1.1710 + }
1.1711 + }
1.1712 + iPointerBuffer->Add(&aPoint);
1.1713 + SignalBufferReady();
1.1714 + }
1.1715 +
1.1716 +void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow* aWindow,TInt aMaxPoints)
1.1717 + {
1.1718 + enum {KPointerMoveBufferSize=32}; // Puts 256 bytes on the stack
1.1719 + if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1.1720 + {
1.1721 + iSignalled=EFalse;
1.1722 + TInt max=Min(aMaxPoints,iPointerBuffer->Count());
1.1723 + TInt buflen=0;
1.1724 + aWindow->WsOwner()->SetReply(max);
1.1725 + TPoint point;
1.1726 + TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
1.1727 + for(TInt index=0;index<max;index++)
1.1728 + {
1.1729 + iPointerBuffer->Remove(&point);
1.1730 + pnts.Append((TUint8 *)&point,sizeof(TPoint));
1.1731 + buflen++;
1.1732 + if (buflen==KPointerMoveBufferSize)
1.1733 + {
1.1734 + CWsClient::ReplyBuf(pnts);
1.1735 + pnts.Zero();
1.1736 + buflen=0;
1.1737 + }
1.1738 + }
1.1739 + if (buflen>0)
1.1740 + {
1.1741 + CWsClient::ReplyBuf(pnts);
1.1742 + }
1.1743 + if (iPointerBuffer->Count())
1.1744 + {
1.1745 + SignalBufferReady();
1.1746 + }
1.1747 + }
1.1748 + }
1.1749 +
1.1750 +void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
1.1751 + {
1.1752 + if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
1.1753 + {
1.1754 + Reset();
1.1755 + }
1.1756 + }
1.1757 +
1.1758 +void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow* aWindow)
1.1759 + {
1.1760 + if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1.1761 + {
1.1762 + Reset();
1.1763 + }
1.1764 + }
1.1765 +
1.1766 +void CWsPointerBuffer::AdjustMaxSizeL()
1.1767 + {
1.1768 + TInt max=0;
1.1769 + CWsPointerBuffer* pb=NULL;
1.1770 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.1771 + {
1.1772 + if (pb->iMaxPoints>max)
1.1773 + {
1.1774 + max=pb->iMaxPoints;
1.1775 + }
1.1776 + }
1.1777 + if (max==0)
1.1778 + {
1.1779 + delete iPointerBuffer;
1.1780 + iPointerBuffer=NULL;
1.1781 + }
1.1782 + else if (!iPointerBuffer)
1.1783 + {
1.1784 + CCirBuf<TPoint>* pointerBuffer=new(ELeave) CCirBuf<TPoint>;
1.1785 + CleanupStack::PushL(pointerBuffer);
1.1786 + pointerBuffer->SetLengthL(max);
1.1787 + CleanupStack::Pop();
1.1788 + iPointerBuffer=pointerBuffer;
1.1789 + }
1.1790 + else
1.1791 + {
1.1792 + iPointerBuffer->SetLengthL(max);
1.1793 + }
1.1794 + }