1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/POINTER.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,867 @@
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 "graphics/pointereventdata.h"
1.34 +
1.35 +TPoint WsPointer::iCurrentPos;
1.36 +TBool WsPointer::iPointerDown;
1.37 +const CWsWindow *WsPointer::iCurrentWindow;
1.38 +const CWsWindow *WsPointer::iActualWinPointerIsOver;
1.39 +const CWsWindow *WsPointer::iGrabWindow;
1.40 +const CWsWindow *WsPointer::iPrevClickWindow;
1.41 +TPoint WsPointer::iPrevClickPos;
1.42 +TTime WsPointer::iPrevClickTime;
1.43 +TPointerEvent::TType WsPointer::iPrevClickEventType;
1.44 +TTimeIntervalMicroSeconds32 WsPointer::iDoubleClickMaxInterval;
1.45 +TInt WsPointer::iDoubleClickMaxDistance;
1.46 +CWsPointerCursor *WsPointer::iCursorSprite;
1.47 +CWsPointerTimer *WsPointer::iRepeatTimer=NULL;
1.48 +TPointerCursorMode WsPointer::iPointerCursorMode=EPointerCursorNormal;
1.49 +CWsWindow *WsPointer::iRepeatWindow=NULL;
1.50 +TRect WsPointer::iRepeatRect;
1.51 +TXYInputType WsPointer::iXyInputType;
1.52 +TUint WsPointer::iLastUnmatchedDown1;
1.53 +TUint WsPointer::iLastUnmatchedDown2;
1.54 +TUint WsPointer::iLastUnmatchedDown3;
1.55 +TBool WsPointer::iTimerQueued;
1.56 +TBool WsPointer::iUpdateRequired;
1.57 +CPeriodic *WsPointer::iPeriodicTimer;
1.58 +CWsRootWindow* WsPointer::iRootWindow;
1.59 +
1.60 +TBool CWsPointerBuffer::iSignalled=EFalse;
1.61 +CWsPointerBuffer *CWsPointerBuffer::iCurrentBuffer=NULL;
1.62 +CCirBuf<TPoint> *CWsPointerBuffer::iPointerBuffer=NULL;
1.63 +TSglQue<CWsPointerBuffer> CWsPointerBuffer::iList(_FOFF(CWsPointerBuffer,iQue));
1.64 +
1.65 +void WsPointer::InitStaticsL()
1.66 + {
1.67 + iRepeatTimer=new(ELeave) CWsPointerTimer();
1.68 + iRepeatTimer->ConstructL();
1.69 + TMachineInfoV1Buf machineInfo;
1.70 + UserHal::MachineInfo(machineInfo);
1.71 + iXyInputType=machineInfo().iXYInputType;
1.72 +
1.73 + iRootWindow = CWsTop::Screen()->RootWindow();
1.74 +
1.75 + iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
1.76 + iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
1.77 + }
1.78 +
1.79 +void WsPointer::DeleteStatics()
1.80 + {
1.81 + SetPointerCursorMode(EPointerCursorNone);
1.82 + UpdatePointerCursor();
1.83 + delete iRepeatTimer;
1.84 + delete iPeriodicTimer;
1.85 + }
1.86 +
1.87 +void WsPointer::SetPointerCursorPos(TPoint aPos)
1.88 + {
1.89 + RestrictPos(aPos,EFalse);
1.90 + iCurrentPos=aPos;
1.91 + ReLogCurrentWindow();
1.92 + UpdatePointerCursor();
1.93 + }
1.94 +
1.95 +void WsPointer::SendEnterExitEvent(TEventCode aType)
1.96 + {
1.97 + if (iCurrentWindow
1.98 + && !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
1.99 + && !iCurrentWindow->ShutDownInProgress())
1.100 + iCurrentWindow->QueueEvent(aType);
1.101 + }
1.102 +
1.103 +void WsPointer::SetCurrentWindow(const CWsWindow *aWin)
1.104 + {
1.105 + if (aWin!=iCurrentWindow)
1.106 + {
1.107 + SendEnterExitEvent(EEventPointerExit);
1.108 + iCurrentWindow=aWin;
1.109 + SendEnterExitEvent(EEventPointerEnter);
1.110 + }
1.111 + }
1.112 +
1.113 +void WsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup *aForceInGroup)
1.114 +//
1.115 +// Relog the current pointer window, can be used to set a new iCurrentPos or when the window layout has changed.
1.116 +// Sets iCurrentPos to aPos and modifys aPos to the relative position within the new current window
1.117 +//
1.118 + {
1.119 + if (iRootWindow)
1.120 + SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,&aPos,&aParentPos,iGrabWindow,iActualWinPointerIsOver
1.121 + ,aForceInGroup));
1.122 + else
1.123 + iCurrentWindow=NULL;
1.124 + }
1.125 +
1.126 +void WsPointer::ReLogCurrentWindow()
1.127 +//
1.128 +// Relog the current pointer window when the window layout has changed.
1.129 +//
1.130 + {
1.131 + if (iCurrentWindow) // NULL iCurrentWindow means pointer is up so don't relog it
1.132 + SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
1.133 + }
1.134 +
1.135 +void WsPointer::ReLogWindow(const CWsWindow *aWin)
1.136 +//
1.137 +// Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
1.138 +// is the current window
1.139 +//
1.140 + {
1.141 + if (aWin==iCurrentWindow)
1.142 + SendEnterExitEvent(EEventPointerEnter);
1.143 + }
1.144 +
1.145 +void WsPointer::UnmatchedDownPurged(TPointerEvent::TType aPointerType, TUint aHandle)
1.146 + {
1.147 + if (aPointerType==TPointerEvent::EButton1Up && iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
1.148 + {
1.149 + if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
1.150 + iGrabWindow=NULL;
1.151 + if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
1.152 + CancelPointerRepeatEventRequest();
1.153 + }
1.154 + switch(aPointerType)
1.155 + {
1.156 + case TPointerEvent::EButton1Up:
1.157 + iLastUnmatchedDown1=aHandle;
1.158 + break;
1.159 + case TPointerEvent::EButton2Up:
1.160 + iLastUnmatchedDown2=aHandle;
1.161 + break;
1.162 + case TPointerEvent::EButton3Up:
1.163 + iLastUnmatchedDown3=aHandle;
1.164 + break;
1.165 + default:;
1.166 + }
1.167 + }
1.168 +
1.169 +void WsPointer::WindowDisconected(const CWsWindow *deletedWindow)
1.170 + {
1.171 + if (iRepeatWindow==deletedWindow)
1.172 + CancelPointerRepeatEventRequest();
1.173 + if (iGrabWindow==deletedWindow)
1.174 + iGrabWindow=NULL;
1.175 + if (iCurrentWindow==deletedWindow)
1.176 + {
1.177 + ReLogCurrentWindow();
1.178 + UpdatePointerCursor();
1.179 + }
1.180 + }
1.181 +
1.182 +TEventQueueWalkRet RemovePointerUpFunc(TAny *aHandle, TWsEvent *aEvent)
1.183 +//
1.184 +// Callback function pointer for up event remove event queue walk
1.185 +//
1.186 + {
1.187 + if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
1.188 + return(EEventQueueWalkDeleteEvent);
1.189 + return(EEventQueueWalkOk);
1.190 + }
1.191 +
1.192 +void WsPointer::ClaimGrab(const CWsWindow *aWindow,TBool aSendUpEvent)
1.193 +//
1.194 +// If the pointer is down claim grab in aWindow as though the down event had gone to this window
1.195 +// also send an up event to the window (if any) that would receive it the pointer was released now
1.196 +//
1.197 + {
1.198 + TInt modState=TWindowServerEvent::GetModifierState();
1.199 + TWsEvent event;
1.200 + TPointerEvent& pointerEvent=*event.Pointer();
1.201 + pointerEvent.iModifiers=modState;
1.202 + pointerEvent.iPosition=iCurrentPos;
1.203 + if (iPointerDown)
1.204 + {
1.205 + if (iCurrentWindow!=aWindow)
1.206 + {
1.207 + if (aSendUpEvent)
1.208 + ProcessEvent(TPointerEvent::EButton1Up,iCurrentPos,modState,NULL,EFalse);
1.209 + else // If up event already in queue purge it
1.210 + {
1.211 + TUint handle=iCurrentWindow->ClientHandle();
1.212 + iCurrentWindow->EventQueue()->WalkEventQueue(&RemovePointerUpFunc,&handle);
1.213 + }
1.214 + iPointerDown=ETrue;
1.215 + if (aWindow->HasPointerGrab())
1.216 + iGrabWindow=aWindow;
1.217 + ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
1.218 + pointerEvent.iType=TPointerEvent::EDrag;
1.219 + ProcessPointerEvent(event);
1.220 + }
1.221 + }
1.222 + else
1.223 + {
1.224 + const CWsWindow *current=iCurrentWindow;
1.225 + iCurrentWindow=aWindow;
1.226 + WS_ASSERT_DEBUG(iGrabWindow==NULL, EWsPanicPointerClaimGrab);
1.227 + iGrabWindow=aWindow; // Force the up event to be sent to aWindow
1.228 + ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
1.229 + pointerEvent.iType=TPointerEvent::EButton1Up;
1.230 + ProcessPointerEvent(event);
1.231 + iGrabWindow=NULL;
1.232 + iCurrentWindow=current;
1.233 + }
1.234 + }
1.235 +
1.236 +TBool WsPointer::CheckDownEventPurged(TPointerEvent::TType aType)
1.237 + {
1.238 + switch(aType)
1.239 + {
1.240 + TUint lastUnmatchedDown;
1.241 + case TPointerEvent::EButton1Up:
1.242 + lastUnmatchedDown=iLastUnmatchedDown1;
1.243 + iLastUnmatchedDown1=0;
1.244 + goto lastUnmatchedDownCheck;
1.245 + case TPointerEvent::EButton2Up:
1.246 + lastUnmatchedDown=iLastUnmatchedDown2;
1.247 + iLastUnmatchedDown2=0;
1.248 + goto lastUnmatchedDownCheck;
1.249 + case TPointerEvent::EButton3Up:
1.250 + lastUnmatchedDown=iLastUnmatchedDown3;
1.251 + iLastUnmatchedDown3=0;
1.252 +lastUnmatchedDownCheck:
1.253 + if (lastUnmatchedDown==iCurrentWindow->ClientHandle())
1.254 + return ETrue; // Don't deliver the event as we've already thrown away the down
1.255 + default: //Should never get to default
1.256 + break;
1.257 + }
1.258 + return EFalse;
1.259 + }
1.260 +
1.261 +TBool WsPointer::QueuePointerEvent(const CWsWindow *aWindow, TWsEvent &aEvent)
1.262 + {
1.263 + CWsClient *client=aWindow->WsOwner();
1.264 + if (client)
1.265 + {
1.266 + CEventQueue *queue=aWindow->EventQueue();
1.267 + aEvent.SetHandle(aWindow->ClientHandle());
1.268 + if (aEvent.Handle()!=0)
1.269 + {
1.270 + if (aEvent.Pointer()->iType==TPointerEvent::EMove || aEvent.Pointer()->iType==TPointerEvent::EDrag)
1.271 + {
1.272 + queue->Wait();
1.273 + const TWsEvent *prev=queue->PeekLastEvent();
1.274 + if (prev!=NULL && prev->Type()==EEventPointer && prev->Handle()==aEvent.Handle() && prev->Pointer()->iType==aEvent.Pointer()->iType)
1.275 + {
1.276 + queue->UpdateLastEvent(aEvent);
1.277 + return EFalse;
1.278 + }
1.279 + queue->Signal();
1.280 + }
1.281 + TWservEventPriorities priority=EEventPriorityLow;
1.282 + switch (aEvent.Pointer()->iType)
1.283 + {
1.284 + case TPointerEvent::EButton1Up:
1.285 + case TPointerEvent::EButton2Up:
1.286 + case TPointerEvent::EButton3Up:
1.287 + if (CheckDownEventPurged(aEvent.Pointer()->iType))
1.288 + return ETrue;
1.289 + if (queue->CheckRoom())
1.290 + {
1.291 + if (CheckDownEventPurged(aEvent.Pointer()->iType))
1.292 + return ETrue;
1.293 + }
1.294 + /*Fall Through if an event was not purged*/
1.295 + case TPointerEvent::EButton1Down:
1.296 + case TPointerEvent::EButton2Down:
1.297 + case TPointerEvent::EButton3Down:
1.298 + priority=EEventPriorityHigh;
1.299 + break;
1.300 + default:;
1.301 + }
1.302 + queue->QueueEvent(aEvent,priority);
1.303 + }
1.304 + }
1.305 + return EFalse;
1.306 + }
1.307 +
1.308 +void WsPointer::ProcessForegroundCheck()
1.309 + {
1.310 + CWsWindowGroup *group=iCurrentWindow->WinGroup();
1.311 + if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
1.312 + group->SetOrdinalPosition(0);
1.313 + }
1.314 +
1.315 +void WsPointer::ProcessPointerEvent(TWsEvent& aEvent)
1.316 + {
1.317 + if (iCurrentWindow && iCurrentWindow!=iRootWindow)
1.318 + {
1.319 + aEvent.SetType(EEventPointer);
1.320 + aEvent.SetTimeNow();
1.321 + TPointerEvent::TType type=aEvent.Pointer()->iType;
1.322 + switch(type)
1.323 + {
1.324 + //TUint lastUnmatchedDown;
1.325 + case TPointerEvent::EButton1Down:
1.326 + ProcessForegroundCheck();
1.327 + /*Fall Through*/
1.328 + case TPointerEvent::EButton2Down:
1.329 + case TPointerEvent::EButton3Down:
1.330 + {
1.331 + TPoint& pos=aEvent.Pointer()->iPosition;
1.332 + if (iCurrentWindow==iPrevClickWindow &&
1.333 + type==iPrevClickEventType &&
1.334 + (Abs(pos.iX-iPrevClickPos.iX)+Abs(pos.iY-iPrevClickPos.iY))<iDoubleClickMaxDistance &&
1.335 + aEvent.Time()<(iPrevClickTime+iDoubleClickMaxInterval))
1.336 + {
1.337 + aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
1.338 + iPrevClickWindow=NULL; // Set to NULL to block a double double click
1.339 + }
1.340 + else
1.341 + iPrevClickWindow=iCurrentWindow;
1.342 + iPrevClickEventType=type;
1.343 + iPrevClickPos=pos;
1.344 + iPrevClickTime=aEvent.Time();
1.345 + }
1.346 + break;
1.347 + default:
1.348 + break;
1.349 + }
1.350 + if (iRepeatWindow)
1.351 + {
1.352 + if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
1.353 + return;
1.354 + CancelPointerRepeatEventRequest();
1.355 + }
1.356 + if (QueuePointerEvent(iCurrentWindow, aEvent))
1.357 + return;
1.358 + if (iCurrentWindow->DragDropCapture())
1.359 + {
1.360 + aEvent.SetType(EEventDragDrop);
1.361 + QueuePointerEvent(iActualWinPointerIsOver, aEvent);
1.362 + }
1.363 + }
1.364 + }
1.365 +
1.366 +TInt PointerTimerCallBack(TAny *)
1.367 + {
1.368 + WsPointer::TimerExpired();
1.369 + return(KErrNone);
1.370 + }
1.371 +
1.372 +void WsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
1.373 + {
1.374 + CScreen* screen = iRootWindow->Screen();
1.375 + WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
1.376 +#if defined(__WINS__)
1.377 + if (aWithinDrawableArea)
1.378 + {
1.379 + if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
1.380 + {
1.381 + return; //Not in the drawable area so user may be trying to click on facia button.
1.382 + }
1.383 + }
1.384 +#endif
1.385 + TRect validRect=screen->GetPointerCursorArea(screen->ScreenSizeMode());
1.386 + if (aPos.iX<validRect.iTl.iX)
1.387 + aPos.iX=validRect.iTl.iX;
1.388 + else if (aPos.iX>=validRect.iBr.iX)
1.389 + aPos.iX=validRect.iBr.iX-1;
1.390 + if (aPos.iY<validRect.iTl.iY)
1.391 + aPos.iY=validRect.iTl.iY;
1.392 + else if (aPos.iY>=validRect.iBr.iY)
1.393 + aPos.iY=validRect.iBr.iY-1;
1.394 + }
1.395 +
1.396 +#if defined(__WINS__)
1.397 +TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
1.398 +#else
1.399 +TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent)
1.400 +#endif
1.401 + {
1.402 +#if defined(__WINS__)
1.403 + WS_ASSERT_DEBUG(TRawEvent::EPointerMove==1, EWsPanicRawEventsTypeChanged);
1.404 + WS_ASSERT_DEBUG(TRawEvent::EPointerMove+1==TRawEvent::EPointerSwitchOn, EWsPanicRawEventsTypeChanged);
1.405 + WS_ASSERT_DEBUG(TRawEvent::EPointerSwitchOn+8==TRawEvent::EButton1Down, EWsPanicRawEventsTypeChanged);
1.406 + WS_ASSERT_DEBUG(TRawEvent::EButton1Down+5==TRawEvent::EButton3Up, EWsPanicRawEventsTypeChanged);
1.407 +#endif
1.408 + TRawEvent::TType type=aRawEvent.Type();
1.409 + if (type<TRawEvent::EPointerMove || (type>TRawEvent::EPointerSwitchOn && type<TRawEvent::EButton1Down)
1.410 + || type>TRawEvent::EButton3Up)
1.411 + return ETrue;
1.412 + if (!XyInput())
1.413 + return EFalse;
1.414 + TPoint xy=aRawEvent.Pos();
1.415 + if (DeltaMouse())
1.416 + {
1.417 + #if defined(__WINS__)
1.418 + if (aFromHardware)
1.419 + return EFalse;
1.420 + #endif
1.421 + if (type==TRawEvent::EPointerMove)
1.422 + {
1.423 + xy+=iCurrentPos;
1.424 + RestrictPos(xy);
1.425 + }
1.426 + else
1.427 + xy=iCurrentPos;
1.428 + }
1.429 + else
1.430 + {
1.431 + CScreen* screen=iRootWindow->Screen();
1.432 + #if !defined(__WINS__)
1.433 + TSize screenSize=screen->ScreenDevice()->SizeInPixels()-TSize(1,1); //This is in the current rotation
1.434 + switch (screen->Orientation())
1.435 + {
1.436 + case CFbsBitGc::EGraphicsOrientationRotated90:
1.437 + xy.SetXY(xy.iY,screenSize.iHeight-xy.iX);
1.438 + break;
1.439 + case CFbsBitGc::EGraphicsOrientationRotated180:
1.440 + xy=-(xy-screenSize);
1.441 + break;
1.442 + case CFbsBitGc::EGraphicsOrientationRotated270:
1.443 + xy.SetXY(screenSize.iWidth-xy.iY,xy.iX);
1.444 + break;
1.445 + default:; //To stop warning
1.446 + }
1.447 + #endif
1.448 + // Move the raw event position by shifting it by Origin and scale
1.449 + xy=screen->PhysicalToLogical(xy);
1.450 + RestrictPos(xy);
1.451 + }
1.452 + aRawEvent.Set(type,xy.iX,xy.iY);
1.453 + return ETrue;
1.454 + }
1.455 +
1.456 +void WsPointer::ProcessEvent(TPointerEvent::TType aType, const TPoint &aPos, TUint aModifiers
1.457 + ,const CWsWindowGroup *aForceInGroup,TBool aNatural)
1.458 + {
1.459 + iCurrentPos=aPos;
1.460 + if (aType==TPointerEvent::EMove && !MovesAvailable() && !iPointerDown)
1.461 + return;
1.462 + TPoint pos(iCurrentPos); // We need a non-const TPoint for 'ReLogCurrentWindow'
1.463 + TPoint parPos;
1.464 + ReLogCurrentWindow(pos,parPos,aForceInGroup);
1.465 + TWsEvent event;
1.466 + TPointerEvent& pointerEvent=*event.Pointer();
1.467 + pointerEvent.iModifiers=aModifiers;
1.468 + pointerEvent.iPosition=pos;
1.469 + pointerEvent.iParentPosition=parPos;
1.470 + switch(aType)
1.471 + {
1.472 + case TPointerEvent::EButton1Down:
1.473 + iPointerDown=ETrue;
1.474 + if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
1.475 + iGrabWindow=iCurrentWindow;
1.476 + if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
1.477 + {
1.478 + pointerEvent.iType=TPointerEvent::EMove;
1.479 + ProcessEvent(event);
1.480 + }
1.481 + break;
1.482 + case TPointerEvent::EButton1Up:
1.483 + iPointerDown=EFalse;
1.484 + iGrabWindow=NULL;
1.485 + break;
1.486 + case TPointerEvent::EMove:
1.487 + if (iPointerDown)
1.488 + aType=TPointerEvent::EDrag;
1.489 + break;
1.490 + default:;
1.491 + }
1.492 + pointerEvent.iType=aType;
1.493 + if (aNatural && CClick::IsHandler())
1.494 + {
1.495 + CClick::PointerEvent(iCurrentPos,pointerEvent);
1.496 + TPointerEventData params;
1.497 + params.iVersion=0;
1.498 + params.iCurrentPos=iCurrentPos;
1.499 + params.iPointerEvent.iType = pointerEvent.iType;
1.500 + params.iPointerEvent.iModifiers = pointerEvent.iModifiers;
1.501 + params.iPointerEvent.iPosition = pointerEvent.iPosition;
1.502 + params.iPointerEvent.iParentPosition = pointerEvent.iParentPosition;
1.503 + params.iClientHandle=iCurrentWindow->ClientHandle();
1.504 + params.iWindowOrigin=iCurrentWindow->Origin();
1.505 + CWsWindowGroup* groupWin=iCurrentWindow->WinGroup();
1.506 + params.iWindowGroupId=groupWin ? groupWin->Identifier() : 0;
1.507 + params.iSource=TPointerEventData::EUnspecified;
1.508 + CClick::OtherEvent(EEventPointer,¶ms);
1.509 + }
1.510 + ProcessEvent(event);
1.511 + }
1.512 +
1.513 +void WsPointer::ProcessEvent(TWsEvent& aEvent)
1.514 + {
1.515 + TUint filter=iCurrentWindow->PointerFilter();
1.516 + TPointerEvent::TType type=aEvent.Pointer()->iType;
1.517 + if ((type!=TPointerEvent::EMove || !(filter&EPointerFilterMove)) &&
1.518 + (type!=TPointerEvent::EDrag || !(filter&EPointerFilterDrag)))
1.519 + {
1.520 + TPoint pos=aEvent.Pointer()->iPosition;
1.521 + if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
1.522 + CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
1.523 + else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
1.524 + ProcessPointerEvent(aEvent);
1.525 + }
1.526 + if (!MovesAvailable() && (type==TPointerEvent::EButton1Up || type==TPointerEvent::ESwitchOn))
1.527 + iCurrentWindow=NULL;
1.528 + PointerCursorUpdateCheck();
1.529 + }
1.530 +
1.531 +void WsPointer::TimerExpired()
1.532 + {
1.533 + WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
1.534 + if (iUpdateRequired)
1.535 + {
1.536 + UpdatePointerCursor();
1.537 + iUpdateRequired=EFalse;
1.538 + }
1.539 + else
1.540 + {
1.541 + iTimerQueued=EFalse;
1.542 + iPeriodicTimer->Cancel();
1.543 + }
1.544 + }
1.545 +
1.546 +void WsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
1.547 + {
1.548 + aTime=iDoubleClickMaxInterval;
1.549 + aDistance=iDoubleClickMaxDistance;
1.550 + }
1.551 +
1.552 +void WsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
1.553 + {
1.554 + iDoubleClickMaxInterval=aTime;
1.555 + iDoubleClickMaxDistance=aDistance;
1.556 + }
1.557 +
1.558 +void WsPointer::PointerCursorUpdateCheck()
1.559 + {
1.560 + CWsPointerCursor *sprite=CalculatePointerCursor();
1.561 + if (iCursorSprite || sprite) // If there either was, or is a pointer cursor we need an update
1.562 + {
1.563 + if (!iTimerQueued)
1.564 + {
1.565 + UpdatePointerCursorTo(sprite);
1.566 + iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1.567 + TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
1.568 + TCallBack(PointerTimerCallBack,NULL));
1.569 + iTimerQueued=ETrue;
1.570 + }
1.571 + else
1.572 + iUpdateRequired=ETrue;
1.573 + }
1.574 + }
1.575 +
1.576 +void WsPointer::UpdatePointerCursor()
1.577 + {
1.578 +//__PROFILE_START(3);
1.579 + CWsPointerCursor *sprite=CalculatePointerCursor();
1.580 + UpdatePointerCursorTo(sprite);
1.581 +//__PROFILE_END(3);
1.582 + }
1.583 +
1.584 +void WsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
1.585 + {
1.586 + CScreen* screen=NULL;
1.587 + if (iCursorSprite!=aNewCursor)
1.588 + {
1.589 + if (iCursorSprite)
1.590 + {
1.591 + iCursorSprite->Deactivate();
1.592 + screen=iCursorSprite->Screen(); //This will need changing ##
1.593 + }
1.594 + iCursorSprite=aNewCursor;
1.595 + if (iCursorSprite)
1.596 + {
1.597 + iCursorSprite->SetPos(iCurrentPos);
1.598 + iCursorSprite->Activate();
1.599 + }
1.600 + goto Update;
1.601 + }
1.602 + else if (iCursorSprite)
1.603 + {
1.604 + iCursorSprite->SetPos(iCurrentPos);
1.605 + Update:
1.606 + if (!screen)
1.607 + screen=iCursorSprite->Screen();
1.608 + screen->Update();
1.609 + }
1.610 + }
1.611 +
1.612 +CWsPointerCursor* WsPointer::CalculatePointerCursor()
1.613 + {
1.614 + CWsPointerCursor *sprite=NULL;
1.615 + if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
1.616 + {
1.617 + const CWsWindowBase* window=iCurrentWindow;
1.618 + do {
1.619 + sprite=window->PointerCursor();
1.620 + if (window->WinType()!=EWinTypeClient)
1.621 + break;
1.622 + window=window->BaseParent();
1.623 + } while (!sprite);
1.624 + }
1.625 + if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
1.626 + sprite=CWsClient::DefaultSystemPointerCursor();
1.627 + return sprite;
1.628 + }
1.629 +
1.630 +TEventQueueWalkRet PointerRepeatPurgeFunc(TAny *, TWsEvent *aEvent)
1.631 +//
1.632 +// Callback function for event queue walk
1.633 +//
1.634 + {
1.635 + return(WsPointer::PointerRepeatPurgeCheck(aEvent));
1.636 + }
1.637 +
1.638 +TBool WsPointer::PointerEventRepeatCheck(const TWsEvent *aEvent, TUint32 aHandle)
1.639 +//
1.640 +// Return ETrue if this pointer event is consumed by the pointer repeat
1.641 +//
1.642 + {
1.643 + TPointerEvent *pntEvent=aEvent->Pointer();
1.644 + if (aHandle==iRepeatWindow->ClientHandle() &&
1.645 + (pntEvent->iType==TPointerEvent::EDrag || pntEvent->iType==TPointerEvent::EMove) &&
1.646 + iRepeatRect.Contains(pntEvent->iPosition))
1.647 + return(ETrue);
1.648 + return(EFalse);
1.649 + }
1.650 +
1.651 +TEventQueueWalkRet WsPointer::PointerRepeatPurgeCheck(const TWsEvent *aEvent)
1.652 + {
1.653 + if (iRepeatWindow && aEvent->Type()==EEventPointer)
1.654 + {
1.655 + if (PointerEventRepeatCheck(aEvent,aEvent->Handle()))
1.656 + return(EEventQueueWalkDeleteEvent); // Purge the event as it is a move/drag within the repeat rect
1.657 + CancelPointerRepeatEventRequest();
1.658 + }
1.659 + return(EEventQueueWalkOk);
1.660 + }
1.661 +
1.662 +void WsPointer::RequestPointerRepeatEvent(CWsWindow *aWindow, TTimeIntervalMicroSeconds32 aTime,const TRect &aRect)
1.663 + {
1.664 + CancelPointerRepeatEventRequest();
1.665 + iRepeatWindow=aWindow;
1.666 + iRepeatRect=aRect;
1.667 + iRepeatTimer->After(aTime);
1.668 + aWindow->EventQueue()->WalkEventQueue(&PointerRepeatPurgeFunc,NULL);
1.669 + if (iRepeatWindow && !iRepeatRect.Contains(iCurrentPos-iRepeatWindow->Origin()))
1.670 + CancelPointerRepeatEventRequest();
1.671 + }
1.672 +
1.673 +void WsPointer::CancelPointerRepeatEventRequest()
1.674 + {
1.675 + if (iRepeatWindow)
1.676 + {
1.677 + iRepeatWindow=NULL;
1.678 + iRepeatTimer->Cancel();
1.679 + }
1.680 + }
1.681 +
1.682 +void WsPointer::RepeatTimerCompleted()
1.683 + {
1.684 + TWsEvent event;
1.685 + event.SetType(EEventPointer);
1.686 + event.SetTimeNow();
1.687 + event.Pointer()->iModifiers=TWindowServerEvent::GetModifierState();
1.688 + event.Pointer()->iPosition=iCurrentPos-iRepeatWindow->Origin();
1.689 + event.Pointer()->iParentPosition=iCurrentPos-iRepeatWindow->BaseParent()->Origin();
1.690 + event.Pointer()->iType=TPointerEvent::EButtonRepeat;
1.691 + QueuePointerEvent(iRepeatWindow, event);
1.692 + iRepeatWindow=NULL;
1.693 + }
1.694 +
1.695 +#if defined(__WINS__)
1.696 +void WsPointer::SetXyInputType(TXYInputType aXyInputType)
1.697 + {
1.698 + if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse && !iPointerDown)
1.699 + {
1.700 + iCurrentWindow=NULL;
1.701 + UpdatePointerCursor();
1.702 + }
1.703 + else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer && !iPointerDown)
1.704 + {
1.705 + TPoint pos(iCurrentPos);
1.706 + TPoint parPos;
1.707 + ReLogCurrentWindow(pos,parPos,NULL);
1.708 + UpdatePointerCursor();
1.709 + }
1.710 + iXyInputType=aXyInputType;
1.711 + }
1.712 +#endif
1.713 +
1.714 +//
1.715 +
1.716 +CWsPointerTimer::CWsPointerTimer() : CTimer(EPointerRepeatPriority)
1.717 + {}
1.718 +
1.719 +void CWsPointerTimer::ConstructL()
1.720 + {
1.721 + CTimer::ConstructL();
1.722 + CActiveScheduler::Add(this);
1.723 + }
1.724 +
1.725 +void CWsPointerTimer::RunL()
1.726 + {
1.727 + User::ResetInactivityTime();
1.728 + WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
1.729 + WsPointer::RepeatTimerCompleted();
1.730 + }
1.731 +
1.732 +//
1.733 +
1.734 +CWsPointerBuffer::~CWsPointerBuffer()
1.735 + {
1.736 + if (this == iCurrentBuffer)
1.737 + {
1.738 + // We're about to be destroyed - don't want to be pointed at any more.
1.739 + iCurrentBuffer = NULL;
1.740 + }
1.741 + iList.Remove(*this);
1.742 + }
1.743 +
1.744 +void CWsPointerBuffer::ConnectL(CWsClientWindow *aWindow, TInt aMaxPoints, TUint aFlags)
1.745 + {
1.746 + CWsPointerBuffer *pb=NULL;
1.747 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.748 + if (pb->iWindow==aWindow)
1.749 + User::Leave(KErrInUse);
1.750 + CWsPointerBuffer *pbuf=new(ELeave) CWsPointerBuffer;
1.751 + pbuf->iWindow=aWindow;
1.752 + pbuf->iMaxPoints=aMaxPoints;
1.753 + pbuf->iFlags=aFlags;
1.754 + iList.AddFirst(*pbuf);
1.755 + CleanupStack::PushL(pbuf);
1.756 + AdjustMaxSizeL();
1.757 + CleanupStack::Pop();
1.758 + }
1.759 +
1.760 +void CWsPointerBuffer::Disconnect(CWsClientWindow *aWindow)
1.761 + {
1.762 + CWsPointerBuffer *pb=NULL;
1.763 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.764 + {
1.765 + if (pb->iWindow==aWindow)
1.766 + {
1.767 + delete pb; // Note that the destructor also sets iCurrentBuffer to NULL if it is pointing at pb
1.768 + TRAP_IGNORE(AdjustMaxSizeL()); // Shouldn't fail, but doesn't matter if it does as we simply have a larger buffer than needed
1.769 + break; // from for loop
1.770 + }
1.771 + }
1.772 + }
1.773 +
1.774 +void CWsPointerBuffer::Reset()
1.775 + {
1.776 + iSignalled=EFalse;
1.777 + iPointerBuffer->Reset();
1.778 + }
1.779 +
1.780 +void CWsPointerBuffer::SignalBufferReady()
1.781 + {
1.782 + if (!iSignalled)
1.783 + if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
1.784 + iSignalled=ETrue;
1.785 + }
1.786 +
1.787 +void CWsPointerBuffer::PointerEvent(CWsClientWindow *aWindow,const TPoint &aPoint)
1.788 + {
1.789 + if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
1.790 + {
1.791 + Reset();
1.792 + CWsPointerBuffer *pb=NULL;
1.793 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.794 + {
1.795 + if (pb->iWindow==aWindow)
1.796 + {
1.797 + iCurrentBuffer=pb;
1.798 + break; // from for loop
1.799 + }
1.800 + }
1.801 + }
1.802 + iPointerBuffer->Add(&aPoint);
1.803 + SignalBufferReady();
1.804 + }
1.805 +
1.806 +void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow *aWindow,TInt aMaxPoints)
1.807 + {
1.808 + enum {KPointerMoveBufferSize=32}; // Puts 256 bytes on the stack
1.809 + if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1.810 + {
1.811 + iSignalled=EFalse;
1.812 + TInt max=Min(aMaxPoints,iPointerBuffer->Count());
1.813 + TInt buflen=0;
1.814 + aWindow->WsOwner()->SetReply(max);
1.815 + TPoint point;
1.816 + TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
1.817 + for(TInt index=0;index<max;index++)
1.818 + {
1.819 + iPointerBuffer->Remove(&point);
1.820 + pnts.Append((TUint8 *)&point,sizeof(TPoint));
1.821 + buflen++;
1.822 + if (buflen==KPointerMoveBufferSize)
1.823 + {
1.824 + CWsClient::ReplyBuf(pnts);
1.825 + pnts.Zero();
1.826 + buflen=0;
1.827 + }
1.828 + }
1.829 + if (buflen>0)
1.830 + CWsClient::ReplyBuf(pnts);
1.831 + if (iPointerBuffer->Count())
1.832 + SignalBufferReady();
1.833 + }
1.834 + }
1.835 +
1.836 +void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
1.837 + {
1.838 + if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
1.839 + Reset();
1.840 + }
1.841 +
1.842 +void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow *aWindow)
1.843 + {
1.844 + if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
1.845 + Reset();
1.846 + }
1.847 +
1.848 +void CWsPointerBuffer::AdjustMaxSizeL()
1.849 + {
1.850 + TInt max=0;
1.851 + CWsPointerBuffer *pb=NULL;
1.852 + for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
1.853 + if (pb->iMaxPoints>max)
1.854 + max=pb->iMaxPoints;
1.855 + if (max==0)
1.856 + {
1.857 + delete iPointerBuffer;
1.858 + iPointerBuffer=NULL;
1.859 + }
1.860 + else if (!iPointerBuffer)
1.861 + {
1.862 + CCirBuf<TPoint> *pointerBuffer=new(ELeave) CCirBuf<TPoint>;
1.863 + CleanupStack::PushL(pointerBuffer);
1.864 + pointerBuffer->SetLengthL(max);
1.865 + CleanupStack::Pop();
1.866 + iPointerBuffer=pointerBuffer;
1.867 + }
1.868 + else
1.869 + iPointerBuffer->SetLengthL(max);
1.870 + }