Update contrib.
1 // Copyright (c) 1995-2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
23 #include "windowgroup.h"
30 #include "graphics/pointereventdata.h"
32 TPoint WsPointer::iCurrentPos;
33 TBool WsPointer::iPointerDown;
34 const CWsWindow *WsPointer::iCurrentWindow;
35 const CWsWindow *WsPointer::iActualWinPointerIsOver;
36 const CWsWindow *WsPointer::iGrabWindow;
37 const CWsWindow *WsPointer::iPrevClickWindow;
38 TPoint WsPointer::iPrevClickPos;
39 TTime WsPointer::iPrevClickTime;
40 TPointerEvent::TType WsPointer::iPrevClickEventType;
41 TTimeIntervalMicroSeconds32 WsPointer::iDoubleClickMaxInterval;
42 TInt WsPointer::iDoubleClickMaxDistance;
43 CWsPointerCursor *WsPointer::iCursorSprite;
44 CWsPointerTimer *WsPointer::iRepeatTimer=NULL;
45 TPointerCursorMode WsPointer::iPointerCursorMode=EPointerCursorNormal;
46 CWsWindow *WsPointer::iRepeatWindow=NULL;
47 TRect WsPointer::iRepeatRect;
48 TXYInputType WsPointer::iXyInputType;
49 TUint WsPointer::iLastUnmatchedDown1;
50 TUint WsPointer::iLastUnmatchedDown2;
51 TUint WsPointer::iLastUnmatchedDown3;
52 TBool WsPointer::iTimerQueued;
53 TBool WsPointer::iUpdateRequired;
54 CPeriodic *WsPointer::iPeriodicTimer;
55 CWsRootWindow* WsPointer::iRootWindow;
57 TBool CWsPointerBuffer::iSignalled=EFalse;
58 CWsPointerBuffer *CWsPointerBuffer::iCurrentBuffer=NULL;
59 CCirBuf<TPoint> *CWsPointerBuffer::iPointerBuffer=NULL;
60 TSglQue<CWsPointerBuffer> CWsPointerBuffer::iList(_FOFF(CWsPointerBuffer,iQue));
62 void WsPointer::InitStaticsL()
64 iRepeatTimer=new(ELeave) CWsPointerTimer();
65 iRepeatTimer->ConstructL();
66 TMachineInfoV1Buf machineInfo;
67 UserHal::MachineInfo(machineInfo);
68 iXyInputType=machineInfo().iXYInputType;
70 iRootWindow = CWsTop::Screen()->RootWindow();
72 iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
73 iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
76 void WsPointer::DeleteStatics()
78 SetPointerCursorMode(EPointerCursorNone);
79 UpdatePointerCursor();
81 delete iPeriodicTimer;
84 void WsPointer::SetPointerCursorPos(TPoint aPos)
86 RestrictPos(aPos,EFalse);
89 UpdatePointerCursor();
92 void WsPointer::SendEnterExitEvent(TEventCode aType)
95 && !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
96 && !iCurrentWindow->ShutDownInProgress())
97 iCurrentWindow->QueueEvent(aType);
100 void WsPointer::SetCurrentWindow(const CWsWindow *aWin)
102 if (aWin!=iCurrentWindow)
104 SendEnterExitEvent(EEventPointerExit);
106 SendEnterExitEvent(EEventPointerEnter);
110 void WsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup *aForceInGroup)
112 // Relog the current pointer window, can be used to set a new iCurrentPos or when the window layout has changed.
113 // Sets iCurrentPos to aPos and modifys aPos to the relative position within the new current window
117 SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,&aPos,&aParentPos,iGrabWindow,iActualWinPointerIsOver
123 void WsPointer::ReLogCurrentWindow()
125 // Relog the current pointer window when the window layout has changed.
128 if (iCurrentWindow) // NULL iCurrentWindow means pointer is up so don't relog it
129 SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
132 void WsPointer::ReLogWindow(const CWsWindow *aWin)
134 // Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
135 // is the current window
138 if (aWin==iCurrentWindow)
139 SendEnterExitEvent(EEventPointerEnter);
142 void WsPointer::UnmatchedDownPurged(TPointerEvent::TType aPointerType, TUint aHandle)
144 if (aPointerType==TPointerEvent::EButton1Up && iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
146 if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
148 if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
149 CancelPointerRepeatEventRequest();
153 case TPointerEvent::EButton1Up:
154 iLastUnmatchedDown1=aHandle;
156 case TPointerEvent::EButton2Up:
157 iLastUnmatchedDown2=aHandle;
159 case TPointerEvent::EButton3Up:
160 iLastUnmatchedDown3=aHandle;
166 void WsPointer::WindowDisconected(const CWsWindow *deletedWindow)
168 if (iRepeatWindow==deletedWindow)
169 CancelPointerRepeatEventRequest();
170 if (iGrabWindow==deletedWindow)
172 if (iCurrentWindow==deletedWindow)
174 ReLogCurrentWindow();
175 UpdatePointerCursor();
179 TEventQueueWalkRet RemovePointerUpFunc(TAny *aHandle, TWsEvent *aEvent)
181 // Callback function pointer for up event remove event queue walk
184 if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
185 return(EEventQueueWalkDeleteEvent);
186 return(EEventQueueWalkOk);
189 void WsPointer::ClaimGrab(const CWsWindow *aWindow,TBool aSendUpEvent)
191 // If the pointer is down claim grab in aWindow as though the down event had gone to this window
192 // also send an up event to the window (if any) that would receive it the pointer was released now
195 TInt modState=TWindowServerEvent::GetModifierState();
197 TPointerEvent& pointerEvent=*event.Pointer();
198 pointerEvent.iModifiers=modState;
199 pointerEvent.iPosition=iCurrentPos;
202 if (iCurrentWindow!=aWindow)
205 ProcessEvent(TPointerEvent::EButton1Up,iCurrentPos,modState,NULL,EFalse);
206 else // If up event already in queue purge it
208 TUint handle=iCurrentWindow->ClientHandle();
209 iCurrentWindow->EventQueue()->WalkEventQueue(&RemovePointerUpFunc,&handle);
212 if (aWindow->HasPointerGrab())
214 ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
215 pointerEvent.iType=TPointerEvent::EDrag;
216 ProcessPointerEvent(event);
221 const CWsWindow *current=iCurrentWindow;
222 iCurrentWindow=aWindow;
223 WS_ASSERT_DEBUG(iGrabWindow==NULL, EWsPanicPointerClaimGrab);
224 iGrabWindow=aWindow; // Force the up event to be sent to aWindow
225 ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
226 pointerEvent.iType=TPointerEvent::EButton1Up;
227 ProcessPointerEvent(event);
229 iCurrentWindow=current;
233 TBool WsPointer::CheckDownEventPurged(TPointerEvent::TType aType)
237 TUint lastUnmatchedDown;
238 case TPointerEvent::EButton1Up:
239 lastUnmatchedDown=iLastUnmatchedDown1;
240 iLastUnmatchedDown1=0;
241 goto lastUnmatchedDownCheck;
242 case TPointerEvent::EButton2Up:
243 lastUnmatchedDown=iLastUnmatchedDown2;
244 iLastUnmatchedDown2=0;
245 goto lastUnmatchedDownCheck;
246 case TPointerEvent::EButton3Up:
247 lastUnmatchedDown=iLastUnmatchedDown3;
248 iLastUnmatchedDown3=0;
249 lastUnmatchedDownCheck:
250 if (lastUnmatchedDown==iCurrentWindow->ClientHandle())
251 return ETrue; // Don't deliver the event as we've already thrown away the down
252 default: //Should never get to default
258 TBool WsPointer::QueuePointerEvent(const CWsWindow *aWindow, TWsEvent &aEvent)
260 CWsClient *client=aWindow->WsOwner();
263 CEventQueue *queue=aWindow->EventQueue();
264 aEvent.SetHandle(aWindow->ClientHandle());
265 if (aEvent.Handle()!=0)
267 if (aEvent.Pointer()->iType==TPointerEvent::EMove || aEvent.Pointer()->iType==TPointerEvent::EDrag)
270 const TWsEvent *prev=queue->PeekLastEvent();
271 if (prev!=NULL && prev->Type()==EEventPointer && prev->Handle()==aEvent.Handle() && prev->Pointer()->iType==aEvent.Pointer()->iType)
273 queue->UpdateLastEvent(aEvent);
278 TWservEventPriorities priority=EEventPriorityLow;
279 switch (aEvent.Pointer()->iType)
281 case TPointerEvent::EButton1Up:
282 case TPointerEvent::EButton2Up:
283 case TPointerEvent::EButton3Up:
284 if (CheckDownEventPurged(aEvent.Pointer()->iType))
286 if (queue->CheckRoom())
288 if (CheckDownEventPurged(aEvent.Pointer()->iType))
291 /*Fall Through if an event was not purged*/
292 case TPointerEvent::EButton1Down:
293 case TPointerEvent::EButton2Down:
294 case TPointerEvent::EButton3Down:
295 priority=EEventPriorityHigh;
299 queue->QueueEvent(aEvent,priority);
305 void WsPointer::ProcessForegroundCheck()
307 CWsWindowGroup *group=iCurrentWindow->WinGroup();
308 if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
309 group->SetOrdinalPosition(0);
312 void WsPointer::ProcessPointerEvent(TWsEvent& aEvent)
314 if (iCurrentWindow && iCurrentWindow!=iRootWindow)
316 aEvent.SetType(EEventPointer);
318 TPointerEvent::TType type=aEvent.Pointer()->iType;
321 //TUint lastUnmatchedDown;
322 case TPointerEvent::EButton1Down:
323 ProcessForegroundCheck();
325 case TPointerEvent::EButton2Down:
326 case TPointerEvent::EButton3Down:
328 TPoint& pos=aEvent.Pointer()->iPosition;
329 if (iCurrentWindow==iPrevClickWindow &&
330 type==iPrevClickEventType &&
331 (Abs(pos.iX-iPrevClickPos.iX)+Abs(pos.iY-iPrevClickPos.iY))<iDoubleClickMaxDistance &&
332 aEvent.Time()<(iPrevClickTime+iDoubleClickMaxInterval))
334 aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
335 iPrevClickWindow=NULL; // Set to NULL to block a double double click
338 iPrevClickWindow=iCurrentWindow;
339 iPrevClickEventType=type;
341 iPrevClickTime=aEvent.Time();
349 if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
351 CancelPointerRepeatEventRequest();
353 if (QueuePointerEvent(iCurrentWindow, aEvent))
355 if (iCurrentWindow->DragDropCapture())
357 aEvent.SetType(EEventDragDrop);
358 QueuePointerEvent(iActualWinPointerIsOver, aEvent);
363 TInt PointerTimerCallBack(TAny *)
365 WsPointer::TimerExpired();
369 void WsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
371 CScreen* screen = iRootWindow->Screen();
372 WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
373 #if defined(__WINS__)
374 if (aWithinDrawableArea)
376 if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
378 return; //Not in the drawable area so user may be trying to click on facia button.
382 TRect validRect=screen->GetPointerCursorArea(screen->ScreenSizeMode());
383 if (aPos.iX<validRect.iTl.iX)
384 aPos.iX=validRect.iTl.iX;
385 else if (aPos.iX>=validRect.iBr.iX)
386 aPos.iX=validRect.iBr.iX-1;
387 if (aPos.iY<validRect.iTl.iY)
388 aPos.iY=validRect.iTl.iY;
389 else if (aPos.iY>=validRect.iBr.iY)
390 aPos.iY=validRect.iBr.iY-1;
393 #if defined(__WINS__)
394 TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
396 TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent)
399 #if defined(__WINS__)
400 WS_ASSERT_DEBUG(TRawEvent::EPointerMove==1, EWsPanicRawEventsTypeChanged);
401 WS_ASSERT_DEBUG(TRawEvent::EPointerMove+1==TRawEvent::EPointerSwitchOn, EWsPanicRawEventsTypeChanged);
402 WS_ASSERT_DEBUG(TRawEvent::EPointerSwitchOn+8==TRawEvent::EButton1Down, EWsPanicRawEventsTypeChanged);
403 WS_ASSERT_DEBUG(TRawEvent::EButton1Down+5==TRawEvent::EButton3Up, EWsPanicRawEventsTypeChanged);
405 TRawEvent::TType type=aRawEvent.Type();
406 if (type<TRawEvent::EPointerMove || (type>TRawEvent::EPointerSwitchOn && type<TRawEvent::EButton1Down)
407 || type>TRawEvent::EButton3Up)
411 TPoint xy=aRawEvent.Pos();
414 #if defined(__WINS__)
418 if (type==TRawEvent::EPointerMove)
428 CScreen* screen=iRootWindow->Screen();
429 #if !defined(__WINS__)
430 TSize screenSize=screen->ScreenDevice()->SizeInPixels()-TSize(1,1); //This is in the current rotation
431 switch (screen->Orientation())
433 case CFbsBitGc::EGraphicsOrientationRotated90:
434 xy.SetXY(xy.iY,screenSize.iHeight-xy.iX);
436 case CFbsBitGc::EGraphicsOrientationRotated180:
439 case CFbsBitGc::EGraphicsOrientationRotated270:
440 xy.SetXY(screenSize.iWidth-xy.iY,xy.iX);
442 default:; //To stop warning
445 // Move the raw event position by shifting it by Origin and scale
446 xy=screen->PhysicalToLogical(xy);
449 aRawEvent.Set(type,xy.iX,xy.iY);
453 void WsPointer::ProcessEvent(TPointerEvent::TType aType, const TPoint &aPos, TUint aModifiers
454 ,const CWsWindowGroup *aForceInGroup,TBool aNatural)
457 if (aType==TPointerEvent::EMove && !MovesAvailable() && !iPointerDown)
459 TPoint pos(iCurrentPos); // We need a non-const TPoint for 'ReLogCurrentWindow'
461 ReLogCurrentWindow(pos,parPos,aForceInGroup);
463 TPointerEvent& pointerEvent=*event.Pointer();
464 pointerEvent.iModifiers=aModifiers;
465 pointerEvent.iPosition=pos;
466 pointerEvent.iParentPosition=parPos;
469 case TPointerEvent::EButton1Down:
471 if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
472 iGrabWindow=iCurrentWindow;
473 if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
475 pointerEvent.iType=TPointerEvent::EMove;
479 case TPointerEvent::EButton1Up:
483 case TPointerEvent::EMove:
485 aType=TPointerEvent::EDrag;
489 pointerEvent.iType=aType;
490 if (aNatural && CClick::IsHandler())
492 CClick::PointerEvent(iCurrentPos,pointerEvent);
493 TPointerEventData params;
495 params.iCurrentPos=iCurrentPos;
496 params.iPointerEvent.iType = pointerEvent.iType;
497 params.iPointerEvent.iModifiers = pointerEvent.iModifiers;
498 params.iPointerEvent.iPosition = pointerEvent.iPosition;
499 params.iPointerEvent.iParentPosition = pointerEvent.iParentPosition;
500 params.iClientHandle=iCurrentWindow->ClientHandle();
501 params.iWindowOrigin=iCurrentWindow->Origin();
502 CWsWindowGroup* groupWin=iCurrentWindow->WinGroup();
503 params.iWindowGroupId=groupWin ? groupWin->Identifier() : 0;
504 params.iSource=TPointerEventData::EUnspecified;
505 CClick::OtherEvent(EEventPointer,¶ms);
510 void WsPointer::ProcessEvent(TWsEvent& aEvent)
512 TUint filter=iCurrentWindow->PointerFilter();
513 TPointerEvent::TType type=aEvent.Pointer()->iType;
514 if ((type!=TPointerEvent::EMove || !(filter&EPointerFilterMove)) &&
515 (type!=TPointerEvent::EDrag || !(filter&EPointerFilterDrag)))
517 TPoint pos=aEvent.Pointer()->iPosition;
518 if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
519 CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
520 else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
521 ProcessPointerEvent(aEvent);
523 if (!MovesAvailable() && (type==TPointerEvent::EButton1Up || type==TPointerEvent::ESwitchOn))
525 PointerCursorUpdateCheck();
528 void WsPointer::TimerExpired()
530 WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
533 UpdatePointerCursor();
534 iUpdateRequired=EFalse;
539 iPeriodicTimer->Cancel();
543 void WsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
545 aTime=iDoubleClickMaxInterval;
546 aDistance=iDoubleClickMaxDistance;
549 void WsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
551 iDoubleClickMaxInterval=aTime;
552 iDoubleClickMaxDistance=aDistance;
555 void WsPointer::PointerCursorUpdateCheck()
557 CWsPointerCursor *sprite=CalculatePointerCursor();
558 if (iCursorSprite || sprite) // If there either was, or is a pointer cursor we need an update
562 UpdatePointerCursorTo(sprite);
563 iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
564 TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
565 TCallBack(PointerTimerCallBack,NULL));
569 iUpdateRequired=ETrue;
573 void WsPointer::UpdatePointerCursor()
575 //__PROFILE_START(3);
576 CWsPointerCursor *sprite=CalculatePointerCursor();
577 UpdatePointerCursorTo(sprite);
581 void WsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
583 CScreen* screen=NULL;
584 if (iCursorSprite!=aNewCursor)
588 iCursorSprite->Deactivate();
589 screen=iCursorSprite->Screen(); //This will need changing ##
591 iCursorSprite=aNewCursor;
594 iCursorSprite->SetPos(iCurrentPos);
595 iCursorSprite->Activate();
599 else if (iCursorSprite)
601 iCursorSprite->SetPos(iCurrentPos);
604 screen=iCursorSprite->Screen();
609 CWsPointerCursor* WsPointer::CalculatePointerCursor()
611 CWsPointerCursor *sprite=NULL;
612 if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
614 const CWsWindowBase* window=iCurrentWindow;
616 sprite=window->PointerCursor();
617 if (window->WinType()!=EWinTypeClient)
619 window=window->BaseParent();
622 if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
623 sprite=CWsClient::DefaultSystemPointerCursor();
627 TEventQueueWalkRet PointerRepeatPurgeFunc(TAny *, TWsEvent *aEvent)
629 // Callback function for event queue walk
632 return(WsPointer::PointerRepeatPurgeCheck(aEvent));
635 TBool WsPointer::PointerEventRepeatCheck(const TWsEvent *aEvent, TUint32 aHandle)
637 // Return ETrue if this pointer event is consumed by the pointer repeat
640 TPointerEvent *pntEvent=aEvent->Pointer();
641 if (aHandle==iRepeatWindow->ClientHandle() &&
642 (pntEvent->iType==TPointerEvent::EDrag || pntEvent->iType==TPointerEvent::EMove) &&
643 iRepeatRect.Contains(pntEvent->iPosition))
648 TEventQueueWalkRet WsPointer::PointerRepeatPurgeCheck(const TWsEvent *aEvent)
650 if (iRepeatWindow && aEvent->Type()==EEventPointer)
652 if (PointerEventRepeatCheck(aEvent,aEvent->Handle()))
653 return(EEventQueueWalkDeleteEvent); // Purge the event as it is a move/drag within the repeat rect
654 CancelPointerRepeatEventRequest();
656 return(EEventQueueWalkOk);
659 void WsPointer::RequestPointerRepeatEvent(CWsWindow *aWindow, TTimeIntervalMicroSeconds32 aTime,const TRect &aRect)
661 CancelPointerRepeatEventRequest();
662 iRepeatWindow=aWindow;
664 iRepeatTimer->After(aTime);
665 aWindow->EventQueue()->WalkEventQueue(&PointerRepeatPurgeFunc,NULL);
666 if (iRepeatWindow && !iRepeatRect.Contains(iCurrentPos-iRepeatWindow->Origin()))
667 CancelPointerRepeatEventRequest();
670 void WsPointer::CancelPointerRepeatEventRequest()
675 iRepeatTimer->Cancel();
679 void WsPointer::RepeatTimerCompleted()
682 event.SetType(EEventPointer);
684 event.Pointer()->iModifiers=TWindowServerEvent::GetModifierState();
685 event.Pointer()->iPosition=iCurrentPos-iRepeatWindow->Origin();
686 event.Pointer()->iParentPosition=iCurrentPos-iRepeatWindow->BaseParent()->Origin();
687 event.Pointer()->iType=TPointerEvent::EButtonRepeat;
688 QueuePointerEvent(iRepeatWindow, event);
692 #if defined(__WINS__)
693 void WsPointer::SetXyInputType(TXYInputType aXyInputType)
695 if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse && !iPointerDown)
698 UpdatePointerCursor();
700 else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer && !iPointerDown)
702 TPoint pos(iCurrentPos);
704 ReLogCurrentWindow(pos,parPos,NULL);
705 UpdatePointerCursor();
707 iXyInputType=aXyInputType;
713 CWsPointerTimer::CWsPointerTimer() : CTimer(EPointerRepeatPriority)
716 void CWsPointerTimer::ConstructL()
718 CTimer::ConstructL();
719 CActiveScheduler::Add(this);
722 void CWsPointerTimer::RunL()
724 User::ResetInactivityTime();
725 WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
726 WsPointer::RepeatTimerCompleted();
731 CWsPointerBuffer::~CWsPointerBuffer()
733 if (this == iCurrentBuffer)
735 // We're about to be destroyed - don't want to be pointed at any more.
736 iCurrentBuffer = NULL;
741 void CWsPointerBuffer::ConnectL(CWsClientWindow *aWindow, TInt aMaxPoints, TUint aFlags)
743 CWsPointerBuffer *pb=NULL;
744 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
745 if (pb->iWindow==aWindow)
746 User::Leave(KErrInUse);
747 CWsPointerBuffer *pbuf=new(ELeave) CWsPointerBuffer;
748 pbuf->iWindow=aWindow;
749 pbuf->iMaxPoints=aMaxPoints;
751 iList.AddFirst(*pbuf);
752 CleanupStack::PushL(pbuf);
757 void CWsPointerBuffer::Disconnect(CWsClientWindow *aWindow)
759 CWsPointerBuffer *pb=NULL;
760 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
762 if (pb->iWindow==aWindow)
764 delete pb; // Note that the destructor also sets iCurrentBuffer to NULL if it is pointing at pb
765 TRAP_IGNORE(AdjustMaxSizeL()); // Shouldn't fail, but doesn't matter if it does as we simply have a larger buffer than needed
766 break; // from for loop
771 void CWsPointerBuffer::Reset()
774 iPointerBuffer->Reset();
777 void CWsPointerBuffer::SignalBufferReady()
780 if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
784 void CWsPointerBuffer::PointerEvent(CWsClientWindow *aWindow,const TPoint &aPoint)
786 if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
789 CWsPointerBuffer *pb=NULL;
790 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
792 if (pb->iWindow==aWindow)
795 break; // from for loop
799 iPointerBuffer->Add(&aPoint);
803 void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow *aWindow,TInt aMaxPoints)
805 enum {KPointerMoveBufferSize=32}; // Puts 256 bytes on the stack
806 if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
809 TInt max=Min(aMaxPoints,iPointerBuffer->Count());
811 aWindow->WsOwner()->SetReply(max);
813 TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
814 for(TInt index=0;index<max;index++)
816 iPointerBuffer->Remove(&point);
817 pnts.Append((TUint8 *)&point,sizeof(TPoint));
819 if (buflen==KPointerMoveBufferSize)
821 CWsClient::ReplyBuf(pnts);
827 CWsClient::ReplyBuf(pnts);
828 if (iPointerBuffer->Count())
833 void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
835 if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
839 void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow *aWindow)
841 if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
845 void CWsPointerBuffer::AdjustMaxSizeL()
848 CWsPointerBuffer *pb=NULL;
849 for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
850 if (pb->iMaxPoints>max)
854 delete iPointerBuffer;
857 else if (!iPointerBuffer)
859 CCirBuf<TPoint> *pointerBuffer=new(ELeave) CCirBuf<TPoint>;
860 CleanupStack::PushL(pointerBuffer);
861 pointerBuffer->SetLengthL(max);
863 iPointerBuffer=pointerBuffer;
866 iPointerBuffer->SetLengthL(max);