windowing/windowserver/TClick/CLICK.CPP
author MattD <mattd@symbian.org>
Mon, 08 Feb 2010 15:55:52 +0000
branchNewGraphicsArchitecture
changeset 1 dd8110ac9267
parent 0 5d03bc08d59c
permissions -rw-r--r--
Created NewGraphicsArchitecture branch for graphics work.

// Copyright (c) 2001-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// Test Key Click Plug-In DLL
// 
//

#include <e32std.h>
#include "W32CLICK.H"
#include "CLICK.H"
#include <graphics/pointereventdata.h>
#if defined(__WINS__)
	#include <emulator.h>
	#include "LOGWIN.H"
#endif

#define bufSize 64

GLREF_D struct TSharedMemory GSharedMemory;

class MClickMaker
	{
public:
	virtual void KeyEvent(TEventCode aType,const TKeyEvent& aEvent)=0;
	virtual void PointerEvent(const TPointerEvent& aEvent)=0;
	virtual void OtherEvent(TInt aType,TAny* aParam)=0;
	};

struct TGpWinInfo
	{
	TBool iInUse;
	TInt iServerValue;
	TInt iNumGroups;
	};

struct TGpWinIdData
	{
	TInt iId;
	TInt iClient;
	};

NONSHARABLE_CLASS(TLogClicks) : public MClickMaker
	{
public:
	void StartLoggingL();
	inline TBool IsLogging() {return iLogging;}
	//Pure virtual functions from MClickMaker
	void KeyEvent(TEventCode aType,const TKeyEvent& aEvent);
	void PointerEvent(const TPointerEvent& aEvent);
    void OtherEvent(TInt aType,TAny* aParam=NULL);
private:
	TBool iLogging;
	};

NONSHARABLE_CLASS(CEventClicks) : public CBase, public MClickMaker
	{
	enum {
		EEventBufferSize=32,
		EMaxGroupWinClients=6,
		EMaxGroupWinIdData=12
		};
public:
	void ConstructL();
	TInt Add(TWsEvent* aEvent);
	inline TInt Failed() {return iFailed;}
	inline TInt Events() {return iEventsTested;}
	void Reset();
	void ExtendedPointerEvent(const TPointerEventData& aPointerEvent);
	void ExpectNewWindowGroup(TInt aClient);
	void NewWindowGroup(const TGroupWindowOpenData& aGpWinOpen);
	void CheckGpWinId(TInt aId);
	void ExpectCloseWindowGroup(TInt aId);
	void CloseWindowGroup(TInt aId);
	void ExpectCloseWindow(TWindowCloseData& aCloseWin);
	void CloseWindow(TWindowCloseData& aCloseWin);
	void PointerEventInfo(TPointerEventInfo& aPointerEventInfo);	
	//Pure virtual functions from MClickMaker
	void KeyEvent(TEventCode aType,const TKeyEvent& aEvent);
	void PointerEvent(const TPointerEvent& aEvent);
	void OtherEvent(TInt aType,TAny* aParam=NULL);
private:
	TBool GetEvent(TWsEvent& aEvent);
	void Fail();
	void AddGroupWindowId(TInt aClient,TInt aGpWinId);
	void RemoveGroupWindowId(TInt aGpWinId);
private:
	CCirBuf<TWsEvent> iEventBuffer;
	TInt iFailed;
	TInt iEventsTested;
	TInt iFailedAt;
	TGpWinInfo iGroupWins[EMaxGroupWinClients];
	TGpWinIdData iGroupWinIds[EMaxGroupWinIdData];
	TInt iExpectedEvent;
	TInt iExpectedEventData;
	TInt iExpectedHandle;
	TPointerEventInfo iExpectedPointerEventInfo;
	TInt iLastNewGpWinId;
	TAdvancedPointerEvent iLastPointerEvent;
	TBool iExpectingExtendedPointer;
	};

NONSHARABLE_CLASS(CMyClickMaker) : public CClickMaker
	{
public:
	~CMyClickMaker();
	void ConstructL();
	//Virtual function from CClickMaker
	void KeyEvent(TEventCode aType,const TKeyEvent& aEvent);
	void PointerEvent(const TPointerEvent& aEvent);
	void OtherEvent(TInt aType,TAny* aParam);
	TInt CommandReplyL(TInt aOpcode, TAny *aArgs);
private:
	void LogToWindowL();
private:
	TClickOutputModes iMode;
	MClickMaker* iCurrentClick;
	TLogClicks iLogClicks;
	CEventClicks* iEventClicks;
	};


/*TLogClicks*/

void TLogClicks::StartLoggingL()
	{
#if defined(__WINS__)
	if (!IsLogging())
		{
		CreateLogWinThreadL();
		iLogging=ETrue;
		}
#else
	User::Leave(KErrNotSupported);
#endif
	}

void TLogClicks::KeyEvent(TEventCode aType,const TKeyEvent& aEvent)
	{
	_LIT(KKeyDown,		"KEY DOWN   ");
	_LIT(KKeyUp,		"KEY UP     ");
	_LIT(KKeyEvent,		"KEY EVENT  ");
	_LIT(KKeyRepeat,	"KEY REPEAT ");
	_LIT(KKeyUnknown,	"KEY Unknown");
	_LIT(KKeyDataFormatChar,"'%c',");
	_LIT(KKeyDataFormatCode," Code=%u,");
	_LIT(KKeyDataFormatScan," Scan=%d,");
	_LIT(KKeyDataFormatModRep," Mod=0x%x, Rep=%d");
	TBuf<bufSize> bufPlusZero;
	switch (aType)
		{
	case EEventKey:
		bufPlusZero.Copy(KKeyEvent);
		break;
	case EEventKeyUp:
		bufPlusZero.Copy(KKeyUp);
		break;
	case EEventKeyDown:
		bufPlusZero.Copy(KKeyDown);
		break;
	case EEventKeyRepeat:
		bufPlusZero.Copy(KKeyRepeat);
		break;
	default:
		bufPlusZero.Copy(KKeyUnknown);
		break;
		}
	bufPlusZero.AppendFormat(KKeyDataFormatCode,aEvent.iCode);
	if (aEvent.iCode!=0)
		bufPlusZero.AppendFormat(KKeyDataFormatChar,aEvent.iCode);
	bufPlusZero.AppendFormat(KKeyDataFormatScan,aEvent.iScanCode);
	if (aEvent.iScanCode!=0)
		bufPlusZero.AppendFormat(KKeyDataFormatChar,aEvent.iScanCode);
	bufPlusZero.AppendFormat(KKeyDataFormatModRep,aEvent.iModifiers,aEvent.iRepeats);
	bufPlusZero.ZeroTerminate();
#if defined(__WINS__)
	Emulator::Escape();
	SendMessage(GSharedMemory.iHwnd, WM_USER+EAppendText, (bufPlusZero.Length()+1)*sizeof(TText), (TInt32)(bufPlusZero.Ptr()));
	Emulator::Reenter();
#endif
	}

void TLogClicks::PointerEvent(const TPointerEvent& aEvent)
	{
	_LIT(KButtonDown,	"POINTER DOWN ");
	_LIT(KButtonUp,		"POINTER UP   ");
	_LIT(KButton2Down,	"BUTTON 2 DOWN");
	_LIT(KButton2Up,	"BUTTON 2 UP  ");
	_LIT(KButton3Down,	"BUTTON 3 DOWN");
	_LIT(KButton3Up,	"BUTTON 3 UP  ");
	_LIT(KButtonDrag,	"POINTER DRAG ");
	_LIT(KButtonMove,	"POINTER MOVE ");
	_LIT(KButtonRepeat,	"BUTTON REPEAT");
	_LIT(KSwitchOn,		"POINTER ON   ");
	_LIT(KUnknown,		"PTR Unknown  ");
	//_LIT(KPtrDataFormat," Pos=(%d,%d), ScrPos=(%d,%d), Modifiers=%x");
	_LIT(KPtrDataFormat," Pos=(%d,%d), ScrPos=(%d,%d), Mod=%x");
	TBuf<bufSize> bufPlusZero;
	switch (aEvent.iType)
		{
	case TPointerEvent::EButton1Down:
		bufPlusZero.Copy(KButtonDown);
		break;
	case TPointerEvent::EButton1Up:
		bufPlusZero.Copy(KButtonUp);
		break;
	case TPointerEvent::EButton2Down:
		bufPlusZero.Copy(KButton2Down);
		break;
	case TPointerEvent::EButton2Up:
		bufPlusZero.Copy(KButton2Up);
		break;
	case TPointerEvent::EButton3Down:
		bufPlusZero.Copy(KButton3Down);
		break;
	case TPointerEvent::EButton3Up:
		bufPlusZero.Copy(KButton3Up);
		break;
	case TPointerEvent::EDrag:
		bufPlusZero.Copy(KButtonDrag);
		break;
	case TPointerEvent::EMove:
		bufPlusZero.Copy(KButtonMove);
		break;
	case TPointerEvent::EButtonRepeat:
		bufPlusZero.Copy(KButtonRepeat);
		break;
	case TPointerEvent::ESwitchOn:
		bufPlusZero.Copy(KSwitchOn);
		break;
	default:
		bufPlusZero.Copy(KUnknown);
		break;
		}
	bufPlusZero.AppendFormat(KPtrDataFormat,aEvent.iPosition.iX,aEvent.iPosition.iY
																,aEvent.iParentPosition.iX,aEvent.iParentPosition.iY,aEvent.iModifiers);
	bufPlusZero.ZeroTerminate();
#if defined(__WINS__)
	Emulator::Escape();
	SendMessage(GSharedMemory.iHwnd, WM_USER+EAppendText, (bufPlusZero.Length()+1)*sizeof(TText), (TInt32)(bufPlusZero.Ptr()));
	Emulator::Reenter();
#endif
	}

void TLogClicks::OtherEvent(TInt aType,TAny* aParam)
	{
	_LIT(KPointer,"POINTER EVENT Ver=%d, ScrPos=(%d,%d), WinClientHandle=0x%x, WinOrigin=%d, WinGpId=%d");
	_LIT(KScreenDeviceChanged,"SCREEN DEVICE CHANGED EVENT, Mode=%d");
	_LIT(KGroupWindowOpen,"GROUP WINDOW OPEN EVENT WinGpId=%d, Client=%d, NumWinGps=%d");
	_LIT(KGroupWindowClose,"GROUP WINDOW CLOSE EVENT WinGpId=%d");
	_LIT(KWindowClose,"WINDOW CLOSE EVENT Client=%d, WinGpId=%d");
	_LIT(KEventUnknown,	"EVENT Unknown");
	TBuf<bufSize> bufPlusZero;
	switch (aType)
		{
	case EEventPointer:
		{
		TPointerEventData& data=*static_cast<TPointerEventData*>(aParam);
		bufPlusZero.Format(KPointer,data.iVersion,data.iCurrentPos.iX,data.iCurrentPos.iY,data.iClientHandle
										,data.iWindowOrigin.iX,data.iWindowOrigin.iY,data.iWindowGroupId);
		}
		break;
	case EEventScreenDeviceChanged:
		{
		TClickMakerData& data=*static_cast<TClickMakerData*>(aParam);
		bufPlusZero.Format(KScreenDeviceChanged,data.screenDeviceMode);
		}
		break;
	case EEventGroupWindowOpen:
		{
		TGroupWindowOpenData& data=*static_cast<TGroupWindowOpenData*>(aParam);
		bufPlusZero.Format(KGroupWindowOpen,data.iIdentifier,data.iClient,data.iNumClientWindowGroups);
		}
		break;
	case EEventGroupWindowClose:
		bufPlusZero.Format(KGroupWindowClose,reinterpret_cast<TInt&>(aParam));
		break;
	case EEventWindowClose:
		{
		TWindowCloseData& data=*static_cast<TWindowCloseData*>(aParam);
		bufPlusZero.Format(KWindowClose,data.iClientHandle,data.iWindowGroupId);
		}
		break;
	default:
		bufPlusZero.Copy(KEventUnknown);
		break;
		}
	bufPlusZero.ZeroTerminate();
#if defined(__WINS__)
	Emulator::Escape();
	SendMessage(GSharedMemory.iHwnd, WM_USER+EAppendText, (bufPlusZero.Length()+1)*sizeof(TText), (TInt32)(bufPlusZero.Ptr()));
	Emulator::Reenter();
#endif
	}

/*CEventClicks*/

void CEventClicks::ConstructL()
	{
	iEventBuffer.SetLengthL(EEventBufferSize);
	}

TInt CEventClicks::Add(TWsEvent* aEvent)
	{
	return (iEventBuffer.Add(aEvent) ? KErrNone:KErrOverflow);
	}

void CEventClicks::Reset()
	{
	iFailed=EFalse;
	iEventsTested=0;
	iExpectedEvent=0;
	iLastNewGpWinId=0;
	}

void CEventClicks::ExtendedPointerEvent(const TPointerEventData& aPointerEvent)
	{
	if (!iExpectingExtendedPointer)
		{
		Fail();
		return;
		}
	iExpectingExtendedPointer=EFalse;
	TBool match=ETrue;
	if (0!=aPointerEvent.iVersion)
		match=EFalse;
	if (iLastPointerEvent.iType!=aPointerEvent.iPointerEvent.iType)
		match=EFalse;
	if (iLastPointerEvent.iModifiers!=aPointerEvent.iPointerEvent.iModifiers)
		match=EFalse;
	if (iLastPointerEvent.iPosition!=aPointerEvent.iPointerEvent.iPosition)
		match=EFalse;
	if (iLastPointerEvent.iParentPosition!=aPointerEvent.iCurrentPos)
		match=EFalse;
	if (TPointerEventData::EUnspecified!=aPointerEvent.iSource)
		match=EFalse;
	if (iExpectedPointerEventInfo.iClientHandle && iExpectedPointerEventInfo.iWinGpId>0)
		{
		if (iLastPointerEvent.iParentPosition-iExpectedPointerEventInfo.iParentOrigin
										!=aPointerEvent.iPointerEvent.iParentPosition)
			match=EFalse;
		if (iExpectedPointerEventInfo.iParentOrigin+iExpectedPointerEventInfo.iWinOrigin
															!=aPointerEvent.iWindowOrigin)
			match=EFalse;
		if (iExpectedPointerEventInfo.iClientHandle!=aPointerEvent.iClientHandle)
			match=EFalse;
		if (iExpectedPointerEventInfo.iWinGpId!=aPointerEvent.iWindowGroupId)
			match=EFalse;
		}
	if (!match)
		Fail();
	}

void CEventClicks::ExpectNewWindowGroup(TInt aClient)
	{
	if (iExpectedEvent>0)
		Fail();
	iExpectedEvent=EEventGroupWindowOpen;
	iExpectedEventData=aClient;
	}

void CEventClicks::NewWindowGroup(const TGroupWindowOpenData& aGpWinOpen)
	{
	iLastNewGpWinId=aGpWinOpen.iIdentifier;
	if (iExpectedEvent!=EEventGroupWindowOpen)
		{
		Fail();
		return;
		}
	iExpectedEvent=0;
	if (iExpectedEventData>=EMaxGroupWinClients)
		return;
	AddGroupWindowId(iExpectedEventData,iLastNewGpWinId);
	TGpWinInfo& gpWinInfo=iGroupWins[iExpectedEventData];
	if (gpWinInfo.iInUse)
		{
		if (aGpWinOpen.iClient!=gpWinInfo.iServerValue)
			Fail();
		else 
			{
			if (aGpWinOpen.iNumClientWindowGroups!=gpWinInfo.iNumGroups)
				Fail();
			++gpWinInfo.iNumGroups;
			}
		}
	else
		{
		gpWinInfo.iInUse=ETrue;
		gpWinInfo.iServerValue=aGpWinOpen.iClient;
		gpWinInfo.iNumGroups=aGpWinOpen.iNumClientWindowGroups+1;
		}
	}

void CEventClicks::CheckGpWinId(TInt aId)
	{
	if (iLastNewGpWinId!=aId)
		Fail();
	}

void CEventClicks::ExpectCloseWindowGroup(TInt aId)
	{
	if (iExpectedEvent>0)
		Fail();
	iExpectedEvent=EEventGroupWindowClose;
	iExpectedEventData=aId;
	}

void CEventClicks::CloseWindowGroup(TInt aId)
	{
	if (iExpectedEvent!=EEventGroupWindowClose || iExpectedEventData!=aId)
		Fail();
	if (iExpectedEvent==EEventGroupWindowClose)
		iExpectedEvent=0;
	RemoveGroupWindowId(aId);
	}

void CEventClicks::ExpectCloseWindow(TWindowCloseData& aCloseWin)
	{
	if (iExpectedEvent>0)
		Fail();
	iExpectedEvent=EEventWindowClose;
	iExpectedEventData=aCloseWin.iWindowGroupId;
	iExpectedHandle=aCloseWin.iClientHandle;
	}

void CEventClicks::CloseWindow(TWindowCloseData& aCloseWin)
	{
	if (iExpectedEvent!=EEventWindowClose || iExpectedEventData!=aCloseWin.iWindowGroupId || iExpectedHandle!=aCloseWin.iClientHandle)
		Fail();
	if (iExpectedEvent==EEventWindowClose)
		iExpectedEvent=0;
	}

void CEventClicks::PointerEventInfo(TPointerEventInfo& aPointerEventInfo)
	{
	iExpectedPointerEventInfo=aPointerEventInfo;
	}

TBool CEventClicks::GetEvent(TWsEvent& aEvent)
	{
	++iEventsTested;
	if (iEventBuffer.Remove(&aEvent)<1)
		{
		Fail();
		return ETrue;
		}
	return EFalse;
	}

void CEventClicks::Fail()
	{
	if (!iFailed)
		{
		iFailed=iEventsTested;
		}
	}

void CEventClicks::AddGroupWindowId(TInt aClient,TInt aGpWinId)
	{
	TInt ii=0;
	while (ii<EMaxGroupWinIdData && iGroupWinIds[ii].iId>0)
		++ii;
	if (ii<EMaxGroupWinIdData)
		{
		iGroupWinIds[ii].iId=aGpWinId;
		iGroupWinIds[ii].iClient=aClient;
		}
	}

void CEventClicks::RemoveGroupWindowId(TInt aGpWinId)
	{
	TInt ii=0;
	while (ii<EMaxGroupWinIdData && iGroupWinIds[ii].iId!=aGpWinId)
		++ii;
	if (ii<EMaxGroupWinIdData)
		{
		--iGroupWins[iGroupWinIds[ii].iClient].iNumGroups;
		iGroupWinIds[ii].iId=0;
		}
	}

#define MODIFIER_FLAGS_TO_IGNOR EModifierNumLock
#pragma warning(disable : 4245)		//'initializing' : conversion from 'int' to 'unsigned int', signed/unsigned mismatch
void CEventClicks::KeyEvent(TEventCode aType,const TKeyEvent& aEvent)
	{
	TBool pass;
	TEventCode eType=aType;
	switch (aType)
		{
	case EEventKey:
		pass=(aEvent.iRepeats==0);
		break;
	case EEventKeyUp:
	case EEventKeyDown:
		pass=(aEvent.iCode==0) && (aEvent.iRepeats==0);
		break;
	case EEventKeyRepeat:
		pass=(aEvent.iRepeats>0);
		eType=EEventKey;
		break;
	default:
		pass=EFalse;
		}
	if (!pass)
		{
		Fail();
		return;
		}
	TWsEvent eEvent;
	if (GetEvent(eEvent))
		return;
	if (eEvent.Type()!=eType)
		{
		Fail();
		return;
		}
	TKeyEvent keyEvent=*eEvent.Key();
	TUint mask=~(EModifierAutorepeatable|MODIFIER_FLAGS_TO_IGNOR);
	if (keyEvent.iCode!=aEvent.iCode || (keyEvent.iModifiers&mask)!=(aEvent.iModifiers&mask) || keyEvent.iScanCode!=aEvent.iScanCode
																						|| (keyEvent.iRepeats==0)!=(aEvent.iRepeats==0))
		{
		Fail();
		return;
		}
	}

void CEventClicks::PointerEvent(const TPointerEvent& aEvent)
	{
	// Click events are now all advanced events so in order to test the modifier bits
	// appropriately we need to copy them as TAdvancedPointerEvent not TPointerEvent
	if(!aEvent.IsAdvancedPointerEvent())
		{
		Fail();
		return;
		}
	iLastPointerEvent=*aEvent.AdvancedPointerEvent();
		
	if (iExpectingExtendedPointer)
		Fail();
	else
		iExpectingExtendedPointer=ETrue;
	TWsEvent eEvent;
	if (GetEvent(eEvent))
		return;
	TAdvancedPointerEvent pEvent=*eEvent.Pointer();
	TUint mask=~(MODIFIER_FLAGS_TO_IGNOR);
	if (pEvent.iType!=aEvent.iType || (pEvent.iModifiers&mask)!=(aEvent.iModifiers&mask)
												|| pEvent.iPosition!=aEvent.iPosition || pEvent.iParentPosition!=aEvent.iParentPosition)
		{
		Fail();
		return;
		}
	}

void CEventClicks::OtherEvent(TInt aType,TAny* aParam)
	{
	TBool pass=ETrue;
	if (aType!=EEventPointer)
		++iEventsTested;
	switch (aType)
		{
	case EEventPointer:
		ExtendedPointerEvent(*static_cast<TPointerEventData*>(aParam));
		break;
	case EEventScreenDeviceChanged:
		break;
	case EEventGroupWindowOpen:
		NewWindowGroup(*static_cast<TGroupWindowOpenData*>(aParam));
		break;
	case EEventGroupWindowClose:
		CloseWindowGroup(reinterpret_cast<TInt>(aParam));
		break;
	case EEventWindowClose:
		CloseWindow(*static_cast<TWindowCloseData*>(aParam));
		break;
	default:
		pass=EFalse;
		}
	if (!pass)
		Fail();
	//GetEvent() is not call here because CWsGroupWindow::EnableScreenChangeEvents() could not be 
	//been called.This mean that no EEventScreenDeviceChanged will be put in the client queue.
	//Instead this event will be always passed to the click plugin if this is present.
	}
#pragma warning(default : 4245)


/*CMyClickMaker*/

CMyClickMaker::~CMyClickMaker()
	{
	delete iEventClicks;
	}

void CMyClickMaker::ConstructL()
	{
	iMode=EClickNone;
	iEventClicks=new(ELeave) CEventClicks();
	iEventClicks->ConstructL();
	}

void CMyClickMaker::KeyEvent(TEventCode aType,const TKeyEvent& aEvent)
	{
	if (iCurrentClick)
		iCurrentClick->KeyEvent(aType,aEvent);
	}

void CMyClickMaker::PointerEvent(const TPointerEvent& aEvent)
	{
	if (iCurrentClick)
		iCurrentClick->PointerEvent(aEvent);
	}

void CMyClickMaker::OtherEvent(TInt aType,TAny* aParam)
	{
	if (iCurrentClick)
		iCurrentClick->OtherEvent(aType,aParam);
	}

TInt CMyClickMaker::CommandReplyL(TInt aOpcode,TAny* aArgs)
	{
	switch (aOpcode)
		{
	case EClickCommandToggleOutput:
		switch (iMode)
			{
		case EClickNone:
			LogToWindowL();
			break;
		case EClickCheck:
		case EClickToWindow:
			iMode=EClickNone;
			iCurrentClick=NULL;
			break;
			}
		break;
	case EClickCommandSetOutput:
		iMode=*STATIC_CAST(TClickOutputModes*,aArgs);
		switch (iMode)
			{
		case EClickNone:
			iCurrentClick=NULL;
			break;
		case EClickCheck:
			iCurrentClick=iEventClicks;
			iEventClicks->Reset();
			break;
		case EClickToWindow:
			LogToWindowL();
			break;
			}
		break;
	case EClickEventAdd:
		return iEventClicks->Add(STATIC_CAST(TWsEvent*,aArgs));
	case EClickFailed:
		return iEventClicks->Failed();
	case EClickEvents:
		return iEventClicks->Events();
	case EClickReset:
		iEventClicks->Reset();
		break;
	case EClickCreateGroupWin:
		iEventClicks->ExpectNewWindowGroup(*static_cast<TInt*>(aArgs));
		break;
	case EClickCheckGpWinId:
		iEventClicks->CheckGpWinId(*static_cast<TInt*>(aArgs));
		break;
	case EClickCloseGroupWin:
		iEventClicks->ExpectCloseWindowGroup(*static_cast<TInt*>(aArgs));
		break;
	case EClickCloseWin:
		iEventClicks->ExpectCloseWindow(*static_cast<TWindowCloseData*>(aArgs));
		break;
	case EClickPointerEvent:
		iEventClicks->PointerEventInfo(*static_cast<TPointerEventInfo*>(aArgs));
		break;
	default:;
		}
	return KErrNone;
	}

void CMyClickMaker::LogToWindowL()
	{
	iMode=EClickNone;
	iCurrentClick=NULL;
	iLogClicks.StartLoggingL();
	iMode=EClickToWindow;
	iCurrentClick=&iLogClicks;
	}


EXPORT_C CClickMaker* CreateClickMakerL()
	{
	CMyClickMaker* plugIn=new(ELeave) CMyClickMaker;
	CleanupStack::PushL(plugIn);
	plugIn->ConstructL();
	CleanupStack::Pop(plugIn);
	return plugIn;
	}