sl@0
|
1 |
// Copyright (c) 1995-2010 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
// Pointer functions
|
sl@0
|
15 |
//
|
sl@0
|
16 |
//
|
sl@0
|
17 |
|
sl@0
|
18 |
#include <e32std.h>
|
sl@0
|
19 |
#include <e32hal.h>
|
sl@0
|
20 |
#include "W32CLICK.H"
|
sl@0
|
21 |
#include "pointer.h"
|
sl@0
|
22 |
#include "rootwin.h"
|
sl@0
|
23 |
#include "windowgroup.h"
|
sl@0
|
24 |
#include "KEYCLICK.H"
|
sl@0
|
25 |
#include "ScrDev.H"
|
sl@0
|
26 |
#include "EVENT.H"
|
sl@0
|
27 |
#include "panics.h"
|
sl@0
|
28 |
#include "wstop.h"
|
sl@0
|
29 |
#include "inifile.h"
|
sl@0
|
30 |
#include "graphics/pointereventdata.h"
|
sl@0
|
31 |
|
sl@0
|
32 |
TPoint WsPointer::iCurrentPos;
|
sl@0
|
33 |
TBool WsPointer::iPointerDown;
|
sl@0
|
34 |
const CWsWindow *WsPointer::iCurrentWindow;
|
sl@0
|
35 |
const CWsWindow *WsPointer::iActualWinPointerIsOver;
|
sl@0
|
36 |
const CWsWindow *WsPointer::iGrabWindow;
|
sl@0
|
37 |
const CWsWindow *WsPointer::iPrevClickWindow;
|
sl@0
|
38 |
TPoint WsPointer::iPrevClickPos;
|
sl@0
|
39 |
TTime WsPointer::iPrevClickTime;
|
sl@0
|
40 |
TPointerEvent::TType WsPointer::iPrevClickEventType;
|
sl@0
|
41 |
TTimeIntervalMicroSeconds32 WsPointer::iDoubleClickMaxInterval;
|
sl@0
|
42 |
TInt WsPointer::iDoubleClickMaxDistance;
|
sl@0
|
43 |
CWsPointerCursor *WsPointer::iCursorSprite;
|
sl@0
|
44 |
CWsPointerTimer *WsPointer::iRepeatTimer=NULL;
|
sl@0
|
45 |
TPointerCursorMode WsPointer::iPointerCursorMode=EPointerCursorNormal;
|
sl@0
|
46 |
CWsWindow *WsPointer::iRepeatWindow=NULL;
|
sl@0
|
47 |
TRect WsPointer::iRepeatRect;
|
sl@0
|
48 |
TXYInputType WsPointer::iXyInputType;
|
sl@0
|
49 |
TUint WsPointer::iLastUnmatchedDown1;
|
sl@0
|
50 |
TUint WsPointer::iLastUnmatchedDown2;
|
sl@0
|
51 |
TUint WsPointer::iLastUnmatchedDown3;
|
sl@0
|
52 |
TBool WsPointer::iTimerQueued;
|
sl@0
|
53 |
TBool WsPointer::iUpdateRequired;
|
sl@0
|
54 |
CPeriodic *WsPointer::iPeriodicTimer;
|
sl@0
|
55 |
CWsRootWindow* WsPointer::iRootWindow;
|
sl@0
|
56 |
|
sl@0
|
57 |
TBool CWsPointerBuffer::iSignalled=EFalse;
|
sl@0
|
58 |
CWsPointerBuffer *CWsPointerBuffer::iCurrentBuffer=NULL;
|
sl@0
|
59 |
CCirBuf<TPoint> *CWsPointerBuffer::iPointerBuffer=NULL;
|
sl@0
|
60 |
TSglQue<CWsPointerBuffer> CWsPointerBuffer::iList(_FOFF(CWsPointerBuffer,iQue));
|
sl@0
|
61 |
|
sl@0
|
62 |
void WsPointer::InitStaticsL()
|
sl@0
|
63 |
{
|
sl@0
|
64 |
iRepeatTimer=new(ELeave) CWsPointerTimer();
|
sl@0
|
65 |
iRepeatTimer->ConstructL();
|
sl@0
|
66 |
TMachineInfoV1Buf machineInfo;
|
sl@0
|
67 |
UserHal::MachineInfo(machineInfo);
|
sl@0
|
68 |
iXyInputType=machineInfo().iXYInputType;
|
sl@0
|
69 |
|
sl@0
|
70 |
iRootWindow = CWsTop::Screen()->RootWindow();
|
sl@0
|
71 |
|
sl@0
|
72 |
iCurrentWindow=MovesAvailable() ? iRootWindow : NULL;
|
sl@0
|
73 |
iPeriodicTimer=CPeriodic::NewL(EPointerCursorPriority);
|
sl@0
|
74 |
}
|
sl@0
|
75 |
|
sl@0
|
76 |
void WsPointer::DeleteStatics()
|
sl@0
|
77 |
{
|
sl@0
|
78 |
SetPointerCursorMode(EPointerCursorNone);
|
sl@0
|
79 |
UpdatePointerCursor();
|
sl@0
|
80 |
delete iRepeatTimer;
|
sl@0
|
81 |
delete iPeriodicTimer;
|
sl@0
|
82 |
}
|
sl@0
|
83 |
|
sl@0
|
84 |
void WsPointer::SetPointerCursorPos(TPoint aPos)
|
sl@0
|
85 |
{
|
sl@0
|
86 |
RestrictPos(aPos,EFalse);
|
sl@0
|
87 |
iCurrentPos=aPos;
|
sl@0
|
88 |
ReLogCurrentWindow();
|
sl@0
|
89 |
UpdatePointerCursor();
|
sl@0
|
90 |
}
|
sl@0
|
91 |
|
sl@0
|
92 |
void WsPointer::SendEnterExitEvent(TEventCode aType)
|
sl@0
|
93 |
{
|
sl@0
|
94 |
if (iCurrentWindow
|
sl@0
|
95 |
&& !(iCurrentWindow->PointerFilter()&EPointerFilterEnterExit)
|
sl@0
|
96 |
&& !iCurrentWindow->ShutDownInProgress())
|
sl@0
|
97 |
iCurrentWindow->QueueEvent(aType);
|
sl@0
|
98 |
}
|
sl@0
|
99 |
|
sl@0
|
100 |
void WsPointer::SetCurrentWindow(const CWsWindow *aWin)
|
sl@0
|
101 |
{
|
sl@0
|
102 |
if (aWin!=iCurrentWindow)
|
sl@0
|
103 |
{
|
sl@0
|
104 |
SendEnterExitEvent(EEventPointerExit);
|
sl@0
|
105 |
iCurrentWindow=aWin;
|
sl@0
|
106 |
SendEnterExitEvent(EEventPointerEnter);
|
sl@0
|
107 |
}
|
sl@0
|
108 |
}
|
sl@0
|
109 |
|
sl@0
|
110 |
void WsPointer::ReLogCurrentWindow(TPoint &aPos, TPoint &aParentPos, const CWsWindowGroup *aForceInGroup)
|
sl@0
|
111 |
//
|
sl@0
|
112 |
// Relog the current pointer window, can be used to set a new iCurrentPos or when the window layout has changed.
|
sl@0
|
113 |
// Sets iCurrentPos to aPos and modifys aPos to the relative position within the new current window
|
sl@0
|
114 |
//
|
sl@0
|
115 |
{
|
sl@0
|
116 |
if (iRootWindow)
|
sl@0
|
117 |
SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,&aPos,&aParentPos,iGrabWindow,iActualWinPointerIsOver
|
sl@0
|
118 |
,aForceInGroup));
|
sl@0
|
119 |
else
|
sl@0
|
120 |
iCurrentWindow=NULL;
|
sl@0
|
121 |
}
|
sl@0
|
122 |
|
sl@0
|
123 |
void WsPointer::ReLogCurrentWindow()
|
sl@0
|
124 |
//
|
sl@0
|
125 |
// Relog the current pointer window when the window layout has changed.
|
sl@0
|
126 |
//
|
sl@0
|
127 |
{
|
sl@0
|
128 |
if (iCurrentWindow) // NULL iCurrentWindow means pointer is up so don't relog it
|
sl@0
|
129 |
SetCurrentWindow(iRootWindow->PointerWindow(iCurrentPos,NULL,NULL,iGrabWindow,iActualWinPointerIsOver,NULL));
|
sl@0
|
130 |
}
|
sl@0
|
131 |
|
sl@0
|
132 |
void WsPointer::ReLogWindow(const CWsWindow *aWin)
|
sl@0
|
133 |
//
|
sl@0
|
134 |
// Called when a window has changed it's filter state, will trigger a 'Enter' message if the window
|
sl@0
|
135 |
// is the current window
|
sl@0
|
136 |
//
|
sl@0
|
137 |
{
|
sl@0
|
138 |
if (aWin==iCurrentWindow)
|
sl@0
|
139 |
SendEnterExitEvent(EEventPointerEnter);
|
sl@0
|
140 |
}
|
sl@0
|
141 |
|
sl@0
|
142 |
void WsPointer::UnmatchedDownPurged(TPointerEvent::TType aPointerType, TUint aHandle)
|
sl@0
|
143 |
{
|
sl@0
|
144 |
if (aPointerType==TPointerEvent::EButton1Up && iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
|
sl@0
|
145 |
{
|
sl@0
|
146 |
if (iGrabWindow && iGrabWindow->ClientHandle()==aHandle)
|
sl@0
|
147 |
iGrabWindow=NULL;
|
sl@0
|
148 |
if (iRepeatWindow && iRepeatWindow->ClientHandle()==aHandle)
|
sl@0
|
149 |
CancelPointerRepeatEventRequest();
|
sl@0
|
150 |
}
|
sl@0
|
151 |
switch(aPointerType)
|
sl@0
|
152 |
{
|
sl@0
|
153 |
case TPointerEvent::EButton1Up:
|
sl@0
|
154 |
iLastUnmatchedDown1=aHandle;
|
sl@0
|
155 |
break;
|
sl@0
|
156 |
case TPointerEvent::EButton2Up:
|
sl@0
|
157 |
iLastUnmatchedDown2=aHandle;
|
sl@0
|
158 |
break;
|
sl@0
|
159 |
case TPointerEvent::EButton3Up:
|
sl@0
|
160 |
iLastUnmatchedDown3=aHandle;
|
sl@0
|
161 |
break;
|
sl@0
|
162 |
default:;
|
sl@0
|
163 |
}
|
sl@0
|
164 |
}
|
sl@0
|
165 |
|
sl@0
|
166 |
void WsPointer::WindowDisconected(const CWsWindow *deletedWindow)
|
sl@0
|
167 |
{
|
sl@0
|
168 |
if (iRepeatWindow==deletedWindow)
|
sl@0
|
169 |
CancelPointerRepeatEventRequest();
|
sl@0
|
170 |
if (iGrabWindow==deletedWindow)
|
sl@0
|
171 |
iGrabWindow=NULL;
|
sl@0
|
172 |
if (iCurrentWindow==deletedWindow)
|
sl@0
|
173 |
{
|
sl@0
|
174 |
ReLogCurrentWindow();
|
sl@0
|
175 |
UpdatePointerCursor();
|
sl@0
|
176 |
}
|
sl@0
|
177 |
}
|
sl@0
|
178 |
|
sl@0
|
179 |
TEventQueueWalkRet RemovePointerUpFunc(TAny *aHandle, TWsEvent *aEvent)
|
sl@0
|
180 |
//
|
sl@0
|
181 |
// Callback function pointer for up event remove event queue walk
|
sl@0
|
182 |
//
|
sl@0
|
183 |
{
|
sl@0
|
184 |
if (aEvent->Type()==EEventPointer && aEvent->Pointer()->iType==TPointerEvent::EButton1Up && (*(TUint *)aHandle)==aEvent->Handle())
|
sl@0
|
185 |
return(EEventQueueWalkDeleteEvent);
|
sl@0
|
186 |
return(EEventQueueWalkOk);
|
sl@0
|
187 |
}
|
sl@0
|
188 |
|
sl@0
|
189 |
void WsPointer::ClaimGrab(const CWsWindow *aWindow,TBool aSendUpEvent)
|
sl@0
|
190 |
//
|
sl@0
|
191 |
// If the pointer is down claim grab in aWindow as though the down event had gone to this window
|
sl@0
|
192 |
// also send an up event to the window (if any) that would receive it the pointer was released now
|
sl@0
|
193 |
//
|
sl@0
|
194 |
{
|
sl@0
|
195 |
TInt modState=TWindowServerEvent::GetModifierState();
|
sl@0
|
196 |
TWsEvent event;
|
sl@0
|
197 |
TPointerEvent& pointerEvent=*event.Pointer();
|
sl@0
|
198 |
pointerEvent.iModifiers=modState;
|
sl@0
|
199 |
pointerEvent.iPosition=iCurrentPos;
|
sl@0
|
200 |
if (iPointerDown)
|
sl@0
|
201 |
{
|
sl@0
|
202 |
if (iCurrentWindow!=aWindow)
|
sl@0
|
203 |
{
|
sl@0
|
204 |
if (aSendUpEvent)
|
sl@0
|
205 |
ProcessEvent(TPointerEvent::EButton1Up,iCurrentPos,modState,NULL,EFalse);
|
sl@0
|
206 |
else // If up event already in queue purge it
|
sl@0
|
207 |
{
|
sl@0
|
208 |
TUint handle=iCurrentWindow->ClientHandle();
|
sl@0
|
209 |
iCurrentWindow->EventQueue()->WalkEventQueue(&RemovePointerUpFunc,&handle);
|
sl@0
|
210 |
}
|
sl@0
|
211 |
iPointerDown=ETrue;
|
sl@0
|
212 |
if (aWindow->HasPointerGrab())
|
sl@0
|
213 |
iGrabWindow=aWindow;
|
sl@0
|
214 |
ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
|
sl@0
|
215 |
pointerEvent.iType=TPointerEvent::EDrag;
|
sl@0
|
216 |
ProcessPointerEvent(event);
|
sl@0
|
217 |
}
|
sl@0
|
218 |
}
|
sl@0
|
219 |
else
|
sl@0
|
220 |
{
|
sl@0
|
221 |
const CWsWindow *current=iCurrentWindow;
|
sl@0
|
222 |
iCurrentWindow=aWindow;
|
sl@0
|
223 |
WS_ASSERT_DEBUG(iGrabWindow==NULL, EWsPanicPointerClaimGrab);
|
sl@0
|
224 |
iGrabWindow=aWindow; // Force the up event to be sent to aWindow
|
sl@0
|
225 |
ReLogCurrentWindow(pointerEvent.iPosition,pointerEvent.iParentPosition,NULL);
|
sl@0
|
226 |
pointerEvent.iType=TPointerEvent::EButton1Up;
|
sl@0
|
227 |
ProcessPointerEvent(event);
|
sl@0
|
228 |
iGrabWindow=NULL;
|
sl@0
|
229 |
iCurrentWindow=current;
|
sl@0
|
230 |
}
|
sl@0
|
231 |
}
|
sl@0
|
232 |
|
sl@0
|
233 |
TBool WsPointer::CheckDownEventPurged(TPointerEvent::TType aType)
|
sl@0
|
234 |
{
|
sl@0
|
235 |
switch(aType)
|
sl@0
|
236 |
{
|
sl@0
|
237 |
TUint lastUnmatchedDown;
|
sl@0
|
238 |
case TPointerEvent::EButton1Up:
|
sl@0
|
239 |
lastUnmatchedDown=iLastUnmatchedDown1;
|
sl@0
|
240 |
iLastUnmatchedDown1=0;
|
sl@0
|
241 |
goto lastUnmatchedDownCheck;
|
sl@0
|
242 |
case TPointerEvent::EButton2Up:
|
sl@0
|
243 |
lastUnmatchedDown=iLastUnmatchedDown2;
|
sl@0
|
244 |
iLastUnmatchedDown2=0;
|
sl@0
|
245 |
goto lastUnmatchedDownCheck;
|
sl@0
|
246 |
case TPointerEvent::EButton3Up:
|
sl@0
|
247 |
lastUnmatchedDown=iLastUnmatchedDown3;
|
sl@0
|
248 |
iLastUnmatchedDown3=0;
|
sl@0
|
249 |
lastUnmatchedDownCheck:
|
sl@0
|
250 |
if (lastUnmatchedDown==iCurrentWindow->ClientHandle())
|
sl@0
|
251 |
return ETrue; // Don't deliver the event as we've already thrown away the down
|
sl@0
|
252 |
default: //Should never get to default
|
sl@0
|
253 |
break;
|
sl@0
|
254 |
}
|
sl@0
|
255 |
return EFalse;
|
sl@0
|
256 |
}
|
sl@0
|
257 |
|
sl@0
|
258 |
TBool WsPointer::QueuePointerEvent(const CWsWindow *aWindow, TWsEvent &aEvent)
|
sl@0
|
259 |
{
|
sl@0
|
260 |
CWsClient *client=aWindow->WsOwner();
|
sl@0
|
261 |
if (client)
|
sl@0
|
262 |
{
|
sl@0
|
263 |
CEventQueue *queue=aWindow->EventQueue();
|
sl@0
|
264 |
aEvent.SetHandle(aWindow->ClientHandle());
|
sl@0
|
265 |
if (aEvent.Handle()!=0)
|
sl@0
|
266 |
{
|
sl@0
|
267 |
if (aEvent.Pointer()->iType==TPointerEvent::EMove || aEvent.Pointer()->iType==TPointerEvent::EDrag)
|
sl@0
|
268 |
{
|
sl@0
|
269 |
queue->Wait();
|
sl@0
|
270 |
const TWsEvent *prev=queue->PeekLastEvent();
|
sl@0
|
271 |
if (prev!=NULL && prev->Type()==EEventPointer && prev->Handle()==aEvent.Handle() && prev->Pointer()->iType==aEvent.Pointer()->iType)
|
sl@0
|
272 |
{
|
sl@0
|
273 |
queue->UpdateLastEvent(aEvent);
|
sl@0
|
274 |
return EFalse;
|
sl@0
|
275 |
}
|
sl@0
|
276 |
queue->Signal();
|
sl@0
|
277 |
}
|
sl@0
|
278 |
TWservEventPriorities priority=EEventPriorityLow;
|
sl@0
|
279 |
switch (aEvent.Pointer()->iType)
|
sl@0
|
280 |
{
|
sl@0
|
281 |
case TPointerEvent::EButton1Up:
|
sl@0
|
282 |
case TPointerEvent::EButton2Up:
|
sl@0
|
283 |
case TPointerEvent::EButton3Up:
|
sl@0
|
284 |
if (CheckDownEventPurged(aEvent.Pointer()->iType))
|
sl@0
|
285 |
return ETrue;
|
sl@0
|
286 |
if (queue->CheckRoom())
|
sl@0
|
287 |
{
|
sl@0
|
288 |
if (CheckDownEventPurged(aEvent.Pointer()->iType))
|
sl@0
|
289 |
return ETrue;
|
sl@0
|
290 |
}
|
sl@0
|
291 |
/*Fall Through if an event was not purged*/
|
sl@0
|
292 |
case TPointerEvent::EButton1Down:
|
sl@0
|
293 |
case TPointerEvent::EButton2Down:
|
sl@0
|
294 |
case TPointerEvent::EButton3Down:
|
sl@0
|
295 |
priority=EEventPriorityHigh;
|
sl@0
|
296 |
break;
|
sl@0
|
297 |
default:;
|
sl@0
|
298 |
}
|
sl@0
|
299 |
queue->QueueEvent(aEvent,priority);
|
sl@0
|
300 |
}
|
sl@0
|
301 |
}
|
sl@0
|
302 |
return EFalse;
|
sl@0
|
303 |
}
|
sl@0
|
304 |
|
sl@0
|
305 |
void WsPointer::ProcessForegroundCheck()
|
sl@0
|
306 |
{
|
sl@0
|
307 |
CWsWindowGroup *group=iCurrentWindow->WinGroup();
|
sl@0
|
308 |
if (group->iFlags&CWsWindowGroup::EGroupFlagAutoForeground)
|
sl@0
|
309 |
group->SetOrdinalPosition(0);
|
sl@0
|
310 |
}
|
sl@0
|
311 |
|
sl@0
|
312 |
void WsPointer::ProcessPointerEvent(TWsEvent& aEvent)
|
sl@0
|
313 |
{
|
sl@0
|
314 |
if (iCurrentWindow && iCurrentWindow!=iRootWindow)
|
sl@0
|
315 |
{
|
sl@0
|
316 |
aEvent.SetType(EEventPointer);
|
sl@0
|
317 |
aEvent.SetTimeNow();
|
sl@0
|
318 |
TPointerEvent::TType type=aEvent.Pointer()->iType;
|
sl@0
|
319 |
switch(type)
|
sl@0
|
320 |
{
|
sl@0
|
321 |
//TUint lastUnmatchedDown;
|
sl@0
|
322 |
case TPointerEvent::EButton1Down:
|
sl@0
|
323 |
ProcessForegroundCheck();
|
sl@0
|
324 |
/*Fall Through*/
|
sl@0
|
325 |
case TPointerEvent::EButton2Down:
|
sl@0
|
326 |
case TPointerEvent::EButton3Down:
|
sl@0
|
327 |
{
|
sl@0
|
328 |
TPoint& pos=aEvent.Pointer()->iPosition;
|
sl@0
|
329 |
if (iCurrentWindow==iPrevClickWindow &&
|
sl@0
|
330 |
type==iPrevClickEventType &&
|
sl@0
|
331 |
(Abs(pos.iX-iPrevClickPos.iX)+Abs(pos.iY-iPrevClickPos.iY))<iDoubleClickMaxDistance &&
|
sl@0
|
332 |
aEvent.Time()<(iPrevClickTime+iDoubleClickMaxInterval))
|
sl@0
|
333 |
{
|
sl@0
|
334 |
aEvent.Pointer()->iModifiers|=EModifierDoubleClick;
|
sl@0
|
335 |
iPrevClickWindow=NULL; // Set to NULL to block a double double click
|
sl@0
|
336 |
}
|
sl@0
|
337 |
else
|
sl@0
|
338 |
iPrevClickWindow=iCurrentWindow;
|
sl@0
|
339 |
iPrevClickEventType=type;
|
sl@0
|
340 |
iPrevClickPos=pos;
|
sl@0
|
341 |
iPrevClickTime=aEvent.Time();
|
sl@0
|
342 |
}
|
sl@0
|
343 |
break;
|
sl@0
|
344 |
default:
|
sl@0
|
345 |
break;
|
sl@0
|
346 |
}
|
sl@0
|
347 |
if (iRepeatWindow)
|
sl@0
|
348 |
{
|
sl@0
|
349 |
if (PointerEventRepeatCheck(&aEvent, iCurrentWindow->ClientHandle()))
|
sl@0
|
350 |
return;
|
sl@0
|
351 |
CancelPointerRepeatEventRequest();
|
sl@0
|
352 |
}
|
sl@0
|
353 |
if (QueuePointerEvent(iCurrentWindow, aEvent))
|
sl@0
|
354 |
return;
|
sl@0
|
355 |
if (iCurrentWindow->DragDropCapture())
|
sl@0
|
356 |
{
|
sl@0
|
357 |
aEvent.SetType(EEventDragDrop);
|
sl@0
|
358 |
QueuePointerEvent(iActualWinPointerIsOver, aEvent);
|
sl@0
|
359 |
}
|
sl@0
|
360 |
}
|
sl@0
|
361 |
}
|
sl@0
|
362 |
|
sl@0
|
363 |
TInt PointerTimerCallBack(TAny *)
|
sl@0
|
364 |
{
|
sl@0
|
365 |
WsPointer::TimerExpired();
|
sl@0
|
366 |
return(KErrNone);
|
sl@0
|
367 |
}
|
sl@0
|
368 |
|
sl@0
|
369 |
void WsPointer::RestrictPos(TPoint& aPos,TBool aWithinDrawableArea/*=ETrue*/)
|
sl@0
|
370 |
{
|
sl@0
|
371 |
CScreen* screen = iRootWindow->Screen();
|
sl@0
|
372 |
WS_ASSERT_DEBUG(screen->IsValidScreenSizeMode(screen->ScreenSizeMode()), EWsPanicInvalidScreenSizeMode);
|
sl@0
|
373 |
#if defined(__WINS__)
|
sl@0
|
374 |
if (aWithinDrawableArea)
|
sl@0
|
375 |
{
|
sl@0
|
376 |
if (!DeltaMouse() && !TRect(screen->DrawableArea()).Contains(aPos))
|
sl@0
|
377 |
{
|
sl@0
|
378 |
return; //Not in the drawable area so user may be trying to click on facia button.
|
sl@0
|
379 |
}
|
sl@0
|
380 |
}
|
sl@0
|
381 |
#endif
|
sl@0
|
382 |
TRect validRect=screen->GetPointerCursorArea(screen->ScreenSizeMode());
|
sl@0
|
383 |
if (aPos.iX<validRect.iTl.iX)
|
sl@0
|
384 |
aPos.iX=validRect.iTl.iX;
|
sl@0
|
385 |
else if (aPos.iX>=validRect.iBr.iX)
|
sl@0
|
386 |
aPos.iX=validRect.iBr.iX-1;
|
sl@0
|
387 |
if (aPos.iY<validRect.iTl.iY)
|
sl@0
|
388 |
aPos.iY=validRect.iTl.iY;
|
sl@0
|
389 |
else if (aPos.iY>=validRect.iBr.iY)
|
sl@0
|
390 |
aPos.iY=validRect.iBr.iY-1;
|
sl@0
|
391 |
}
|
sl@0
|
392 |
|
sl@0
|
393 |
#if defined(__WINS__)
|
sl@0
|
394 |
TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent,TBool aFromHardware/*=EFlase*/)
|
sl@0
|
395 |
#else
|
sl@0
|
396 |
TBool WsPointer::PreProcessEvent(TRawEvent &aRawEvent)
|
sl@0
|
397 |
#endif
|
sl@0
|
398 |
{
|
sl@0
|
399 |
#if defined(__WINS__)
|
sl@0
|
400 |
WS_ASSERT_DEBUG(TRawEvent::EPointerMove==1, EWsPanicRawEventsTypeChanged);
|
sl@0
|
401 |
WS_ASSERT_DEBUG(TRawEvent::EPointerMove+1==TRawEvent::EPointerSwitchOn, EWsPanicRawEventsTypeChanged);
|
sl@0
|
402 |
WS_ASSERT_DEBUG(TRawEvent::EPointerSwitchOn+8==TRawEvent::EButton1Down, EWsPanicRawEventsTypeChanged);
|
sl@0
|
403 |
WS_ASSERT_DEBUG(TRawEvent::EButton1Down+5==TRawEvent::EButton3Up, EWsPanicRawEventsTypeChanged);
|
sl@0
|
404 |
#endif
|
sl@0
|
405 |
TRawEvent::TType type=aRawEvent.Type();
|
sl@0
|
406 |
if (type<TRawEvent::EPointerMove || (type>TRawEvent::EPointerSwitchOn && type<TRawEvent::EButton1Down)
|
sl@0
|
407 |
|| type>TRawEvent::EButton3Up)
|
sl@0
|
408 |
return ETrue;
|
sl@0
|
409 |
if (!XyInput())
|
sl@0
|
410 |
return EFalse;
|
sl@0
|
411 |
TPoint xy=aRawEvent.Pos();
|
sl@0
|
412 |
if (DeltaMouse())
|
sl@0
|
413 |
{
|
sl@0
|
414 |
#if defined(__WINS__)
|
sl@0
|
415 |
if (aFromHardware)
|
sl@0
|
416 |
return EFalse;
|
sl@0
|
417 |
#endif
|
sl@0
|
418 |
if (type==TRawEvent::EPointerMove)
|
sl@0
|
419 |
{
|
sl@0
|
420 |
xy+=iCurrentPos;
|
sl@0
|
421 |
RestrictPos(xy);
|
sl@0
|
422 |
}
|
sl@0
|
423 |
else
|
sl@0
|
424 |
xy=iCurrentPos;
|
sl@0
|
425 |
}
|
sl@0
|
426 |
else
|
sl@0
|
427 |
{
|
sl@0
|
428 |
CScreen* screen=iRootWindow->Screen();
|
sl@0
|
429 |
#if !defined(__WINS__)
|
sl@0
|
430 |
TSize screenSize=screen->ScreenDevice()->SizeInPixels()-TSize(1,1); //This is in the current rotation
|
sl@0
|
431 |
switch (screen->Orientation())
|
sl@0
|
432 |
{
|
sl@0
|
433 |
case CFbsBitGc::EGraphicsOrientationRotated90:
|
sl@0
|
434 |
xy.SetXY(xy.iY,screenSize.iHeight-xy.iX);
|
sl@0
|
435 |
break;
|
sl@0
|
436 |
case CFbsBitGc::EGraphicsOrientationRotated180:
|
sl@0
|
437 |
xy=-(xy-screenSize);
|
sl@0
|
438 |
break;
|
sl@0
|
439 |
case CFbsBitGc::EGraphicsOrientationRotated270:
|
sl@0
|
440 |
xy.SetXY(screenSize.iWidth-xy.iY,xy.iX);
|
sl@0
|
441 |
break;
|
sl@0
|
442 |
default:; //To stop warning
|
sl@0
|
443 |
}
|
sl@0
|
444 |
#endif
|
sl@0
|
445 |
// Move the raw event position by shifting it by Origin and scale
|
sl@0
|
446 |
xy=screen->PhysicalToLogical(xy);
|
sl@0
|
447 |
RestrictPos(xy);
|
sl@0
|
448 |
}
|
sl@0
|
449 |
aRawEvent.Set(type,xy.iX,xy.iY);
|
sl@0
|
450 |
return ETrue;
|
sl@0
|
451 |
}
|
sl@0
|
452 |
|
sl@0
|
453 |
void WsPointer::ProcessEvent(TPointerEvent::TType aType, const TPoint &aPos, TUint aModifiers
|
sl@0
|
454 |
,const CWsWindowGroup *aForceInGroup,TBool aNatural)
|
sl@0
|
455 |
{
|
sl@0
|
456 |
iCurrentPos=aPos;
|
sl@0
|
457 |
if (aType==TPointerEvent::EMove && !MovesAvailable() && !iPointerDown)
|
sl@0
|
458 |
return;
|
sl@0
|
459 |
TPoint pos(iCurrentPos); // We need a non-const TPoint for 'ReLogCurrentWindow'
|
sl@0
|
460 |
TPoint parPos;
|
sl@0
|
461 |
ReLogCurrentWindow(pos,parPos,aForceInGroup);
|
sl@0
|
462 |
TWsEvent event;
|
sl@0
|
463 |
TPointerEvent& pointerEvent=*event.Pointer();
|
sl@0
|
464 |
pointerEvent.iModifiers=aModifiers;
|
sl@0
|
465 |
pointerEvent.iPosition=pos;
|
sl@0
|
466 |
pointerEvent.iParentPosition=parPos;
|
sl@0
|
467 |
switch(aType)
|
sl@0
|
468 |
{
|
sl@0
|
469 |
case TPointerEvent::EButton1Down:
|
sl@0
|
470 |
iPointerDown=ETrue;
|
sl@0
|
471 |
if (iGrabWindow==NULL && iCurrentWindow->HasPointerGrab())
|
sl@0
|
472 |
iGrabWindow=iCurrentWindow;
|
sl@0
|
473 |
if (!MovesAvailable() && iCurrentWindow->PointerFilter()&EPointerGenerateSimulatedMove)
|
sl@0
|
474 |
{
|
sl@0
|
475 |
pointerEvent.iType=TPointerEvent::EMove;
|
sl@0
|
476 |
ProcessEvent(event);
|
sl@0
|
477 |
}
|
sl@0
|
478 |
break;
|
sl@0
|
479 |
case TPointerEvent::EButton1Up:
|
sl@0
|
480 |
iPointerDown=EFalse;
|
sl@0
|
481 |
iGrabWindow=NULL;
|
sl@0
|
482 |
break;
|
sl@0
|
483 |
case TPointerEvent::EMove:
|
sl@0
|
484 |
if (iPointerDown)
|
sl@0
|
485 |
aType=TPointerEvent::EDrag;
|
sl@0
|
486 |
break;
|
sl@0
|
487 |
default:;
|
sl@0
|
488 |
}
|
sl@0
|
489 |
pointerEvent.iType=aType;
|
sl@0
|
490 |
if (aNatural && CClick::IsHandler())
|
sl@0
|
491 |
{
|
sl@0
|
492 |
CClick::PointerEvent(iCurrentPos,pointerEvent);
|
sl@0
|
493 |
TPointerEventData params;
|
sl@0
|
494 |
params.iVersion=0;
|
sl@0
|
495 |
params.iCurrentPos=iCurrentPos;
|
sl@0
|
496 |
params.iPointerEvent.iType = pointerEvent.iType;
|
sl@0
|
497 |
params.iPointerEvent.iModifiers = pointerEvent.iModifiers;
|
sl@0
|
498 |
params.iPointerEvent.iPosition = pointerEvent.iPosition;
|
sl@0
|
499 |
params.iPointerEvent.iParentPosition = pointerEvent.iParentPosition;
|
sl@0
|
500 |
params.iClientHandle=iCurrentWindow->ClientHandle();
|
sl@0
|
501 |
params.iWindowOrigin=iCurrentWindow->Origin();
|
sl@0
|
502 |
CWsWindowGroup* groupWin=iCurrentWindow->WinGroup();
|
sl@0
|
503 |
params.iWindowGroupId=groupWin ? groupWin->Identifier() : 0;
|
sl@0
|
504 |
params.iSource=TPointerEventData::EUnspecified;
|
sl@0
|
505 |
CClick::OtherEvent(EEventPointer,¶ms);
|
sl@0
|
506 |
}
|
sl@0
|
507 |
ProcessEvent(event);
|
sl@0
|
508 |
}
|
sl@0
|
509 |
|
sl@0
|
510 |
void WsPointer::ProcessEvent(TWsEvent& aEvent)
|
sl@0
|
511 |
{
|
sl@0
|
512 |
TUint filter=iCurrentWindow->PointerFilter();
|
sl@0
|
513 |
TPointerEvent::TType type=aEvent.Pointer()->iType;
|
sl@0
|
514 |
if ((type!=TPointerEvent::EMove || !(filter&EPointerFilterMove)) &&
|
sl@0
|
515 |
(type!=TPointerEvent::EDrag || !(filter&EPointerFilterDrag)))
|
sl@0
|
516 |
{
|
sl@0
|
517 |
TPoint pos=aEvent.Pointer()->iPosition;
|
sl@0
|
518 |
if ((type==TPointerEvent::EMove || type==TPointerEvent::EDrag) && iCurrentWindow->UsingPointerBuffer())
|
sl@0
|
519 |
CWsPointerBuffer::PointerEvent((CWsClientWindow *)iCurrentWindow,pos);
|
sl@0
|
520 |
else if (!WsKeyboardEmulator::PointerEvent(type,pos,iCurrentWindow->PointerKeyList()))
|
sl@0
|
521 |
ProcessPointerEvent(aEvent);
|
sl@0
|
522 |
}
|
sl@0
|
523 |
if (!MovesAvailable() && (type==TPointerEvent::EButton1Up || type==TPointerEvent::ESwitchOn))
|
sl@0
|
524 |
iCurrentWindow=NULL;
|
sl@0
|
525 |
PointerCursorUpdateCheck();
|
sl@0
|
526 |
}
|
sl@0
|
527 |
|
sl@0
|
528 |
void WsPointer::TimerExpired()
|
sl@0
|
529 |
{
|
sl@0
|
530 |
WS_ASSERT_DEBUG(iTimerQueued, EWsPanicPointerTimer);
|
sl@0
|
531 |
if (iUpdateRequired)
|
sl@0
|
532 |
{
|
sl@0
|
533 |
UpdatePointerCursor();
|
sl@0
|
534 |
iUpdateRequired=EFalse;
|
sl@0
|
535 |
}
|
sl@0
|
536 |
else
|
sl@0
|
537 |
{
|
sl@0
|
538 |
iTimerQueued=EFalse;
|
sl@0
|
539 |
iPeriodicTimer->Cancel();
|
sl@0
|
540 |
}
|
sl@0
|
541 |
}
|
sl@0
|
542 |
|
sl@0
|
543 |
void WsPointer::GetDoubleClickSettings(TTimeIntervalMicroSeconds32 &aTime, TInt &aDistance)
|
sl@0
|
544 |
{
|
sl@0
|
545 |
aTime=iDoubleClickMaxInterval;
|
sl@0
|
546 |
aDistance=iDoubleClickMaxDistance;
|
sl@0
|
547 |
}
|
sl@0
|
548 |
|
sl@0
|
549 |
void WsPointer::SetDoubleClick(const TTimeIntervalMicroSeconds32 &aTime, TInt aDistance)
|
sl@0
|
550 |
{
|
sl@0
|
551 |
iDoubleClickMaxInterval=aTime;
|
sl@0
|
552 |
iDoubleClickMaxDistance=aDistance;
|
sl@0
|
553 |
}
|
sl@0
|
554 |
|
sl@0
|
555 |
void WsPointer::PointerCursorUpdateCheck()
|
sl@0
|
556 |
{
|
sl@0
|
557 |
CWsPointerCursor *sprite=CalculatePointerCursor();
|
sl@0
|
558 |
if (iCursorSprite || sprite) // If there either was, or is a pointer cursor we need an update
|
sl@0
|
559 |
{
|
sl@0
|
560 |
if (!iTimerQueued)
|
sl@0
|
561 |
{
|
sl@0
|
562 |
UpdatePointerCursorTo(sprite);
|
sl@0
|
563 |
iPeriodicTimer->Start(TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
|
sl@0
|
564 |
TTimeIntervalMicroSeconds32(EPointerUpdateGapInMicroSeconds),
|
sl@0
|
565 |
TCallBack(PointerTimerCallBack,NULL));
|
sl@0
|
566 |
iTimerQueued=ETrue;
|
sl@0
|
567 |
}
|
sl@0
|
568 |
else
|
sl@0
|
569 |
iUpdateRequired=ETrue;
|
sl@0
|
570 |
}
|
sl@0
|
571 |
}
|
sl@0
|
572 |
|
sl@0
|
573 |
void WsPointer::UpdatePointerCursor()
|
sl@0
|
574 |
{
|
sl@0
|
575 |
//__PROFILE_START(3);
|
sl@0
|
576 |
CWsPointerCursor *sprite=CalculatePointerCursor();
|
sl@0
|
577 |
UpdatePointerCursorTo(sprite);
|
sl@0
|
578 |
//__PROFILE_END(3);
|
sl@0
|
579 |
}
|
sl@0
|
580 |
|
sl@0
|
581 |
void WsPointer::UpdatePointerCursorTo(CWsPointerCursor* aNewCursor)
|
sl@0
|
582 |
{
|
sl@0
|
583 |
CScreen* screen=NULL;
|
sl@0
|
584 |
if (iCursorSprite!=aNewCursor)
|
sl@0
|
585 |
{
|
sl@0
|
586 |
if (iCursorSprite)
|
sl@0
|
587 |
{
|
sl@0
|
588 |
iCursorSprite->Deactivate();
|
sl@0
|
589 |
screen=iCursorSprite->Screen(); //This will need changing ##
|
sl@0
|
590 |
}
|
sl@0
|
591 |
iCursorSprite=aNewCursor;
|
sl@0
|
592 |
if (iCursorSprite)
|
sl@0
|
593 |
{
|
sl@0
|
594 |
iCursorSprite->SetPos(iCurrentPos);
|
sl@0
|
595 |
iCursorSprite->Activate();
|
sl@0
|
596 |
}
|
sl@0
|
597 |
goto Update;
|
sl@0
|
598 |
}
|
sl@0
|
599 |
else if (iCursorSprite)
|
sl@0
|
600 |
{
|
sl@0
|
601 |
iCursorSprite->SetPos(iCurrentPos);
|
sl@0
|
602 |
Update:
|
sl@0
|
603 |
if (!screen)
|
sl@0
|
604 |
screen=iCursorSprite->Screen();
|
sl@0
|
605 |
screen->Update();
|
sl@0
|
606 |
}
|
sl@0
|
607 |
}
|
sl@0
|
608 |
|
sl@0
|
609 |
CWsPointerCursor* WsPointer::CalculatePointerCursor()
|
sl@0
|
610 |
{
|
sl@0
|
611 |
CWsPointerCursor *sprite=NULL;
|
sl@0
|
612 |
if (iCurrentWindow && (iPointerCursorMode==EPointerCursorNormal || iPointerCursorMode==EPointerCursorWindow))
|
sl@0
|
613 |
{
|
sl@0
|
614 |
const CWsWindowBase* window=iCurrentWindow;
|
sl@0
|
615 |
do {
|
sl@0
|
616 |
sprite=window->PointerCursor();
|
sl@0
|
617 |
if (window->WinType()!=EWinTypeClient)
|
sl@0
|
618 |
break;
|
sl@0
|
619 |
window=window->BaseParent();
|
sl@0
|
620 |
} while (!sprite);
|
sl@0
|
621 |
}
|
sl@0
|
622 |
if (!sprite && iCurrentWindow && (iPointerCursorMode==EPointerCursorFixed || iPointerCursorMode==EPointerCursorNormal))
|
sl@0
|
623 |
sprite=CWsClient::DefaultSystemPointerCursor();
|
sl@0
|
624 |
return sprite;
|
sl@0
|
625 |
}
|
sl@0
|
626 |
|
sl@0
|
627 |
TEventQueueWalkRet PointerRepeatPurgeFunc(TAny *, TWsEvent *aEvent)
|
sl@0
|
628 |
//
|
sl@0
|
629 |
// Callback function for event queue walk
|
sl@0
|
630 |
//
|
sl@0
|
631 |
{
|
sl@0
|
632 |
return(WsPointer::PointerRepeatPurgeCheck(aEvent));
|
sl@0
|
633 |
}
|
sl@0
|
634 |
|
sl@0
|
635 |
TBool WsPointer::PointerEventRepeatCheck(const TWsEvent *aEvent, TUint32 aHandle)
|
sl@0
|
636 |
//
|
sl@0
|
637 |
// Return ETrue if this pointer event is consumed by the pointer repeat
|
sl@0
|
638 |
//
|
sl@0
|
639 |
{
|
sl@0
|
640 |
TPointerEvent *pntEvent=aEvent->Pointer();
|
sl@0
|
641 |
if (aHandle==iRepeatWindow->ClientHandle() &&
|
sl@0
|
642 |
(pntEvent->iType==TPointerEvent::EDrag || pntEvent->iType==TPointerEvent::EMove) &&
|
sl@0
|
643 |
iRepeatRect.Contains(pntEvent->iPosition))
|
sl@0
|
644 |
return(ETrue);
|
sl@0
|
645 |
return(EFalse);
|
sl@0
|
646 |
}
|
sl@0
|
647 |
|
sl@0
|
648 |
TEventQueueWalkRet WsPointer::PointerRepeatPurgeCheck(const TWsEvent *aEvent)
|
sl@0
|
649 |
{
|
sl@0
|
650 |
if (iRepeatWindow && aEvent->Type()==EEventPointer)
|
sl@0
|
651 |
{
|
sl@0
|
652 |
if (PointerEventRepeatCheck(aEvent,aEvent->Handle()))
|
sl@0
|
653 |
return(EEventQueueWalkDeleteEvent); // Purge the event as it is a move/drag within the repeat rect
|
sl@0
|
654 |
CancelPointerRepeatEventRequest();
|
sl@0
|
655 |
}
|
sl@0
|
656 |
return(EEventQueueWalkOk);
|
sl@0
|
657 |
}
|
sl@0
|
658 |
|
sl@0
|
659 |
void WsPointer::RequestPointerRepeatEvent(CWsWindow *aWindow, TTimeIntervalMicroSeconds32 aTime,const TRect &aRect)
|
sl@0
|
660 |
{
|
sl@0
|
661 |
CancelPointerRepeatEventRequest();
|
sl@0
|
662 |
iRepeatWindow=aWindow;
|
sl@0
|
663 |
iRepeatRect=aRect;
|
sl@0
|
664 |
iRepeatTimer->After(aTime);
|
sl@0
|
665 |
aWindow->EventQueue()->WalkEventQueue(&PointerRepeatPurgeFunc,NULL);
|
sl@0
|
666 |
if (iRepeatWindow && !iRepeatRect.Contains(iCurrentPos-iRepeatWindow->Origin()))
|
sl@0
|
667 |
CancelPointerRepeatEventRequest();
|
sl@0
|
668 |
}
|
sl@0
|
669 |
|
sl@0
|
670 |
void WsPointer::CancelPointerRepeatEventRequest()
|
sl@0
|
671 |
{
|
sl@0
|
672 |
if (iRepeatWindow)
|
sl@0
|
673 |
{
|
sl@0
|
674 |
iRepeatWindow=NULL;
|
sl@0
|
675 |
iRepeatTimer->Cancel();
|
sl@0
|
676 |
}
|
sl@0
|
677 |
}
|
sl@0
|
678 |
|
sl@0
|
679 |
void WsPointer::RepeatTimerCompleted()
|
sl@0
|
680 |
{
|
sl@0
|
681 |
TWsEvent event;
|
sl@0
|
682 |
event.SetType(EEventPointer);
|
sl@0
|
683 |
event.SetTimeNow();
|
sl@0
|
684 |
event.Pointer()->iModifiers=TWindowServerEvent::GetModifierState();
|
sl@0
|
685 |
event.Pointer()->iPosition=iCurrentPos-iRepeatWindow->Origin();
|
sl@0
|
686 |
event.Pointer()->iParentPosition=iCurrentPos-iRepeatWindow->BaseParent()->Origin();
|
sl@0
|
687 |
event.Pointer()->iType=TPointerEvent::EButtonRepeat;
|
sl@0
|
688 |
QueuePointerEvent(iRepeatWindow, event);
|
sl@0
|
689 |
iRepeatWindow=NULL;
|
sl@0
|
690 |
}
|
sl@0
|
691 |
|
sl@0
|
692 |
#if defined(__WINS__)
|
sl@0
|
693 |
void WsPointer::SetXyInputType(TXYInputType aXyInputType)
|
sl@0
|
694 |
{
|
sl@0
|
695 |
if (iXyInputType>EXYInputPointer && aXyInputType<EXYInputMouse && !iPointerDown)
|
sl@0
|
696 |
{
|
sl@0
|
697 |
iCurrentWindow=NULL;
|
sl@0
|
698 |
UpdatePointerCursor();
|
sl@0
|
699 |
}
|
sl@0
|
700 |
else if (iXyInputType<EXYInputMouse && aXyInputType>EXYInputPointer && !iPointerDown)
|
sl@0
|
701 |
{
|
sl@0
|
702 |
TPoint pos(iCurrentPos);
|
sl@0
|
703 |
TPoint parPos;
|
sl@0
|
704 |
ReLogCurrentWindow(pos,parPos,NULL);
|
sl@0
|
705 |
UpdatePointerCursor();
|
sl@0
|
706 |
}
|
sl@0
|
707 |
iXyInputType=aXyInputType;
|
sl@0
|
708 |
}
|
sl@0
|
709 |
#endif
|
sl@0
|
710 |
|
sl@0
|
711 |
//
|
sl@0
|
712 |
|
sl@0
|
713 |
CWsPointerTimer::CWsPointerTimer() : CTimer(EPointerRepeatPriority)
|
sl@0
|
714 |
{}
|
sl@0
|
715 |
|
sl@0
|
716 |
void CWsPointerTimer::ConstructL()
|
sl@0
|
717 |
{
|
sl@0
|
718 |
CTimer::ConstructL();
|
sl@0
|
719 |
CActiveScheduler::Add(this);
|
sl@0
|
720 |
}
|
sl@0
|
721 |
|
sl@0
|
722 |
void CWsPointerTimer::RunL()
|
sl@0
|
723 |
{
|
sl@0
|
724 |
User::ResetInactivityTime();
|
sl@0
|
725 |
WS_ASSERT_DEBUG(iStatus.Int()==KErrNone, EWsPanicPointerRepeatTimerStatus);
|
sl@0
|
726 |
WsPointer::RepeatTimerCompleted();
|
sl@0
|
727 |
}
|
sl@0
|
728 |
|
sl@0
|
729 |
//
|
sl@0
|
730 |
|
sl@0
|
731 |
CWsPointerBuffer::~CWsPointerBuffer()
|
sl@0
|
732 |
{
|
sl@0
|
733 |
if (this == iCurrentBuffer)
|
sl@0
|
734 |
{
|
sl@0
|
735 |
// We're about to be destroyed - don't want to be pointed at any more.
|
sl@0
|
736 |
iCurrentBuffer = NULL;
|
sl@0
|
737 |
}
|
sl@0
|
738 |
iList.Remove(*this);
|
sl@0
|
739 |
}
|
sl@0
|
740 |
|
sl@0
|
741 |
void CWsPointerBuffer::ConnectL(CWsClientWindow *aWindow, TInt aMaxPoints, TUint aFlags)
|
sl@0
|
742 |
{
|
sl@0
|
743 |
CWsPointerBuffer *pb=NULL;
|
sl@0
|
744 |
for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
|
sl@0
|
745 |
if (pb->iWindow==aWindow)
|
sl@0
|
746 |
User::Leave(KErrInUse);
|
sl@0
|
747 |
CWsPointerBuffer *pbuf=new(ELeave) CWsPointerBuffer;
|
sl@0
|
748 |
pbuf->iWindow=aWindow;
|
sl@0
|
749 |
pbuf->iMaxPoints=aMaxPoints;
|
sl@0
|
750 |
pbuf->iFlags=aFlags;
|
sl@0
|
751 |
iList.AddFirst(*pbuf);
|
sl@0
|
752 |
CleanupStack::PushL(pbuf);
|
sl@0
|
753 |
AdjustMaxSizeL();
|
sl@0
|
754 |
CleanupStack::Pop();
|
sl@0
|
755 |
}
|
sl@0
|
756 |
|
sl@0
|
757 |
void CWsPointerBuffer::Disconnect(CWsClientWindow *aWindow)
|
sl@0
|
758 |
{
|
sl@0
|
759 |
CWsPointerBuffer *pb=NULL;
|
sl@0
|
760 |
for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
|
sl@0
|
761 |
{
|
sl@0
|
762 |
if (pb->iWindow==aWindow)
|
sl@0
|
763 |
{
|
sl@0
|
764 |
delete pb; // Note that the destructor also sets iCurrentBuffer to NULL if it is pointing at pb
|
sl@0
|
765 |
TRAP_IGNORE(AdjustMaxSizeL()); // Shouldn't fail, but doesn't matter if it does as we simply have a larger buffer than needed
|
sl@0
|
766 |
break; // from for loop
|
sl@0
|
767 |
}
|
sl@0
|
768 |
}
|
sl@0
|
769 |
}
|
sl@0
|
770 |
|
sl@0
|
771 |
void CWsPointerBuffer::Reset()
|
sl@0
|
772 |
{
|
sl@0
|
773 |
iSignalled=EFalse;
|
sl@0
|
774 |
iPointerBuffer->Reset();
|
sl@0
|
775 |
}
|
sl@0
|
776 |
|
sl@0
|
777 |
void CWsPointerBuffer::SignalBufferReady()
|
sl@0
|
778 |
{
|
sl@0
|
779 |
if (!iSignalled)
|
sl@0
|
780 |
if (iCurrentBuffer && iCurrentBuffer->iWindow->QueueEvent(EEventPointerBufferReady))
|
sl@0
|
781 |
iSignalled=ETrue;
|
sl@0
|
782 |
}
|
sl@0
|
783 |
|
sl@0
|
784 |
void CWsPointerBuffer::PointerEvent(CWsClientWindow *aWindow,const TPoint &aPoint)
|
sl@0
|
785 |
{
|
sl@0
|
786 |
if (iCurrentBuffer==NULL || aWindow!=iCurrentBuffer->iWindow)
|
sl@0
|
787 |
{
|
sl@0
|
788 |
Reset();
|
sl@0
|
789 |
CWsPointerBuffer *pb=NULL;
|
sl@0
|
790 |
for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
|
sl@0
|
791 |
{
|
sl@0
|
792 |
if (pb->iWindow==aWindow)
|
sl@0
|
793 |
{
|
sl@0
|
794 |
iCurrentBuffer=pb;
|
sl@0
|
795 |
break; // from for loop
|
sl@0
|
796 |
}
|
sl@0
|
797 |
}
|
sl@0
|
798 |
}
|
sl@0
|
799 |
iPointerBuffer->Add(&aPoint);
|
sl@0
|
800 |
SignalBufferReady();
|
sl@0
|
801 |
}
|
sl@0
|
802 |
|
sl@0
|
803 |
void CWsPointerBuffer::RetrievePointerMoveBuffer(CWsClientWindow *aWindow,TInt aMaxPoints)
|
sl@0
|
804 |
{
|
sl@0
|
805 |
enum {KPointerMoveBufferSize=32}; // Puts 256 bytes on the stack
|
sl@0
|
806 |
if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
|
sl@0
|
807 |
{
|
sl@0
|
808 |
iSignalled=EFalse;
|
sl@0
|
809 |
TInt max=Min(aMaxPoints,iPointerBuffer->Count());
|
sl@0
|
810 |
TInt buflen=0;
|
sl@0
|
811 |
aWindow->WsOwner()->SetReply(max);
|
sl@0
|
812 |
TPoint point;
|
sl@0
|
813 |
TBuf8<KPointerMoveBufferSize*sizeof(TPoint)> pnts;
|
sl@0
|
814 |
for(TInt index=0;index<max;index++)
|
sl@0
|
815 |
{
|
sl@0
|
816 |
iPointerBuffer->Remove(&point);
|
sl@0
|
817 |
pnts.Append((TUint8 *)&point,sizeof(TPoint));
|
sl@0
|
818 |
buflen++;
|
sl@0
|
819 |
if (buflen==KPointerMoveBufferSize)
|
sl@0
|
820 |
{
|
sl@0
|
821 |
CWsClient::ReplyBuf(pnts);
|
sl@0
|
822 |
pnts.Zero();
|
sl@0
|
823 |
buflen=0;
|
sl@0
|
824 |
}
|
sl@0
|
825 |
}
|
sl@0
|
826 |
if (buflen>0)
|
sl@0
|
827 |
CWsClient::ReplyBuf(pnts);
|
sl@0
|
828 |
if (iPointerBuffer->Count())
|
sl@0
|
829 |
SignalBufferReady();
|
sl@0
|
830 |
}
|
sl@0
|
831 |
}
|
sl@0
|
832 |
|
sl@0
|
833 |
void CWsPointerBuffer::DiscardPointerMoveBuffer(TUint aHandle)
|
sl@0
|
834 |
{
|
sl@0
|
835 |
if (iCurrentBuffer && aHandle==iCurrentBuffer->iWindow->ClientHandle())
|
sl@0
|
836 |
Reset();
|
sl@0
|
837 |
}
|
sl@0
|
838 |
|
sl@0
|
839 |
void CWsPointerBuffer::DiscardPointerMoveBuffer(CWsClientWindow *aWindow)
|
sl@0
|
840 |
{
|
sl@0
|
841 |
if (iCurrentBuffer && aWindow==iCurrentBuffer->iWindow)
|
sl@0
|
842 |
Reset();
|
sl@0
|
843 |
}
|
sl@0
|
844 |
|
sl@0
|
845 |
void CWsPointerBuffer::AdjustMaxSizeL()
|
sl@0
|
846 |
{
|
sl@0
|
847 |
TInt max=0;
|
sl@0
|
848 |
CWsPointerBuffer *pb=NULL;
|
sl@0
|
849 |
for(TSglQueIter<CWsPointerBuffer> iter(iList);(pb=iter++)!=NULL;)
|
sl@0
|
850 |
if (pb->iMaxPoints>max)
|
sl@0
|
851 |
max=pb->iMaxPoints;
|
sl@0
|
852 |
if (max==0)
|
sl@0
|
853 |
{
|
sl@0
|
854 |
delete iPointerBuffer;
|
sl@0
|
855 |
iPointerBuffer=NULL;
|
sl@0
|
856 |
}
|
sl@0
|
857 |
else if (!iPointerBuffer)
|
sl@0
|
858 |
{
|
sl@0
|
859 |
CCirBuf<TPoint> *pointerBuffer=new(ELeave) CCirBuf<TPoint>;
|
sl@0
|
860 |
CleanupStack::PushL(pointerBuffer);
|
sl@0
|
861 |
pointerBuffer->SetLengthL(max);
|
sl@0
|
862 |
CleanupStack::Pop();
|
sl@0
|
863 |
iPointerBuffer=pointerBuffer;
|
sl@0
|
864 |
}
|
sl@0
|
865 |
else
|
sl@0
|
866 |
iPointerBuffer->SetLengthL(max);
|
sl@0
|
867 |
}
|