os/graphics/windowing/windowserver/nonnga/SERVER/POINTER.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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,&params);
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
	}