windowing/windowserver/test/HANDANIM.CPP
author William Roberts <williamr@symbian.org>
Tue, 20 Apr 2010 16:24:43 +0100
branchNewGraphicsArchitecture
changeset 34 76efc8f9f7b4
parent 0 5d03bc08d59c
permissions -rw-r--r--
Apply Faisal's first patch from Bug 2354 - First resolve some the the bit rot in graphics MCL to get it to compile, then fix some performance issues in OpenWF

// Copyright (c) 1999-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:
// Anim DLL to deal with handwriting
// 
//

#include "HANDANIM.H"

#define DEFAUlT_LINE_WIDTH 4
#define DEFAUlT_MASK_WIDTH_FACTOR 3
#define DEFAUlT_END_POINT_FACTOR 2
#define BLACK TRgb::Gray2(0)
#define WHITE TRgb::Gray2(1)

#define leavescan_needs_to_see_some_brackets_here	{	}
EXPORT_C CAnimDll *CreateCAnimDllL()
	{
	return(new(ELeave) CHandWritingAnimDll());
	}


/*CHandWritingAnimDll*/

CAnim *CHandWritingAnimDll::CreateInstanceL(TInt /*aType*/)
	{
	return new(ELeave) CHandWritingAnim();
	}


/*CHandWritingAnim*/

CHandWritingAnim::~CHandWritingAnim()
	{
	iFunctions->GetRawEvents(EFalse);
	delete iBitmapDevice;
	delete iMaskBitmapDevice;
	}

void CHandWritingAnim::Activate()
	{
	if (iState==EHwStateDeactive)
		{
		iState=EHwStateInactive;
		iFunctions->GetRawEvents(ETrue);
		}
	}

void CHandWritingAnim::Deactivate()
	{
	if (iState!=EHwStateDeactive)
		{
		iState=EHwStateDeactive;
		iFunctions->GetRawEvents(EFalse);
		iSpriteFunctions->Activate(EFalse);
		ClearSprite();
		}
	}

void CHandWritingAnim::SpriteChangeL(TBool aUsingSeparateMask)
	{
	if (aUsingSeparateMask)
		{
		TSpriteMember *spriteMember=iSpriteFunctions->GetSpriteMember(0);
		iMaskBitmapDevice=CFbsBitmapDevice::NewL(spriteMember->iMaskBitmap);		//If this leaves the error value will be returned to the client side
		}
	else
		{
		delete iMaskBitmapDevice;
		iMaskBitmapDevice=NULL;
		iDrawData.iLineColor=BLACK;		//Must use black ink when there is no mask
		iDrawData.iInitialBitmapColor=WHITE;		//Must have white background when there is no mask
		}
	iIsMask=aUsingSeparateMask;
	}

void CHandWritingAnim::SetDrawData(THandwritingDrawData *aDrawData)
	{
	iDrawData=*aDrawData;
	if (!iIsMask)
		{
		iDrawData.iLineColor=BLACK;		//Must use black ink when there is no mask
		iDrawData.iInitialBitmapColor=WHITE;		//Must have white background when there is no mask
		}
	}

TBool CHandWritingAnim::HandlePointerDown(TPoint aPoint)
	{
	if (iState==EHwStateWaitingMove)
		return EFalse;
	iCurrentDrawPoint=aPoint;
	if (iState==EHwStateInactive)
		{
		iState=EHwStateWaitingMove;
		StartTimer();
		return ETrue;
		}
	iState=EHwStateDrawing;
	DrawPoint();
	UpdateSprite();
	return ETrue;
	}

TBool CHandWritingAnim::HandlePointerMove(TPoint aPoint)
	{
	switch (iState)
		{
	case EHwStateWaitingMove:
		{
		TPoint moved=aPoint-iCurrentDrawPoint;
		if (Abs(moved.iX)<5 && Abs(moved.iY)<5)			//Need to do something with these constants
			return ETrue;
		iSpriteFunctions->Activate(ETrue);
		DrawPoint();
		iState=EHwStateDrawing;
		}
	case EHwStateDrawing:
		break;
	default:
		return EFalse;
		}
	DrawLine(aPoint);
	UpdateSprite();
	return ETrue;
	}

TBool CHandWritingAnim::HandlePointerUp(TPoint aPoint)
	{
	if (iState==EHwStateInactive)
		return EFalse;
	else if (iState==EHwStateWaitingMove)
		{
		TPoint moved=aPoint-iCurrentDrawPoint;
		if (Abs(moved.iX)<5 && Abs(moved.iY)<5)			//Need to do something with these constants
			{
			SendEatenDownEvent();
			return EFalse;
			}
		iSpriteFunctions->Activate(ETrue);
		DrawPoint();
		}
	DrawLine(aPoint);
	DrawPoint();
	UpdateSprite();
	iState=EHwStateWaitingStroke;
	StartTimer();
	return ETrue;
	}

void CHandWritingAnim::DrawPoint()
	{
	iSpriteGc->Activate(iBitmapDevice);
	iSpriteGc->SetPenSize(TSize(iDrawData.iEndPontWidth,iDrawData.iEndPontWidth));
	iSpriteGc->SetPenColor(iDrawData.iLineColor);
	iSpriteGc->Plot(iCurrentDrawPoint);
	if (iMaskBitmapDevice)
		{
		iSpriteGc->Activate(iMaskBitmapDevice);
		iSpriteGc->SetPenSize(TSize(iDrawData.iMaskLineWidth,iDrawData.iMaskLineWidth));
		iSpriteGc->SetPenColor(BLACK);		//Mask must be drawn in black
		iSpriteGc->Plot(iCurrentDrawPoint);
		}
	iPointStore->AddPoint(iCurrentDrawPoint);
	}

void CHandWritingAnim::DrawLine(TPoint aEndPoint)
	{
	iSpriteGc->Activate(iBitmapDevice);
	iSpriteGc->SetPenSize(TSize(iDrawData.iLineWidth,iDrawData.iLineWidth));
	iSpriteGc->SetPenColor(iDrawData.iLineColor);
	iSpriteGc->MoveTo(iCurrentDrawPoint);
	iSpriteGc->DrawLineTo(aEndPoint);
	if (iMaskBitmapDevice)
		{
		iSpriteGc->Activate(iMaskBitmapDevice);
		iSpriteGc->SetPenSize(TSize(iDrawData.iMaskLineWidth,iDrawData.iMaskLineWidth));
		iSpriteGc->SetPenColor(BLACK);		//Mask must be drawn in black
		iSpriteGc->MoveTo(iCurrentDrawPoint);
		iSpriteGc->DrawLineTo(aEndPoint);
		}
	iCurrentDrawPoint=aEndPoint;
	iPointStore->AddPoint(aEndPoint);
	}

void CHandWritingAnim::UpdateSprite()
	{
	TRect drawTo;
	iSpriteGc->RectDrawnTo(drawTo);
	iSpriteFunctions->UpdateMember(0,drawTo,EFalse);
	}

void CHandWritingAnim::StartTimer()
	{
	iFunctions->SetNextInterval(2);
	}

void CHandWritingAnim::SendEatenDownEvent()
	{
	TRawEvent rawEvent;
	rawEvent.Set(TRawEvent::EButton1Down,iCurrentDrawPoint.iX,iCurrentDrawPoint.iY);
	iFunctions->PostRawEvent(rawEvent);
	iState=EHwStateInactive;
	}

void CHandWritingAnim::CharacterFinished()
	{		
	iState=EHwStateInactive;
	iLastGeneratedCharacter=iPointStore->GetChar();
	/*TRawEvent rawEvent;
	rawEvent.Set(TRawEvent::EKeyDown,iLastGeneratedCharacter);
	iFunctions->PostKeyEvent(rawEvent);*/
    TKeyEvent keyEvent;
    keyEvent.iCode=keyEvent.iScanCode=iLastGeneratedCharacter;
    keyEvent.iModifiers=keyEvent.iRepeats=0;
	iFunctions->PostKeyEvent(keyEvent);
	iPointStore->ClearPoints();
	iSpriteFunctions->Activate(EFalse);
	ClearSprite();
	}

void CHandWritingAnim::ClearSprite()
	{
	iSpriteGc->Activate(iBitmapDevice);
	iSpriteGc->SetBrushStyle(CGraphicsContext::ESolidBrush);
	iSpriteGc->SetBrushColor(iDrawData.iInitialBitmapColor);
	iSpriteGc->Clear();
	if (iMaskBitmapDevice)
		{
		iSpriteGc->Activate(iMaskBitmapDevice);
		iSpriteGc->SetBrushColor(WHITE);		//Mask must be cleared in white
		iSpriteGc->Clear();
		}
    TRect drawnTo;
	iSpriteGc->RectDrawnTo(drawnTo);		//Clear the drawnTo rect.
	}

TBool CHandWritingAnim::OfferRawEvent(const TRawEvent &aRawEvent)
	{
	if (iState==EHwStateDeactive)
		return EFalse;
	switch (aRawEvent.Type())
		{
	case TRawEvent::EButton1Down:
		{
		iDownTime.HomeTime();		
		return HandlePointerDown(aRawEvent.Pos());
		}
	case TRawEvent::EPointerMove:
		return HandlePointerMove(aRawEvent.Pos());
	case TRawEvent::EButton1Up:
		return HandlePointerUp(aRawEvent.Pos());
	default:
		return EFalse;
		}
	}

void CHandWritingAnim::ConstructL(TAny *)
	{
	TSpriteMember *spriteMember=iSpriteFunctions->GetSpriteMember(0);
	iIsMask=(spriteMember->iBitmap->Handle() != spriteMember->iMaskBitmap->Handle());
	iBitmapDevice=CFbsBitmapDevice::NewL(spriteMember->iBitmap);
	if (iIsMask)
		iMaskBitmapDevice=CFbsBitmapDevice::NewL(spriteMember->iMaskBitmap);
	iState=EHwStateDeactive;
	iSpriteGc=CFbsBitGc::NewL();
	iSpriteGc->Reset();
	iDrawData.iLineColor=BLACK;
	iDrawData.iInitialBitmapColor=WHITE;
	iDrawData.iLineWidth=DEFAUlT_LINE_WIDTH;
	iDrawData.iMaskLineWidth=DEFAUlT_MASK_WIDTH_FACTOR*DEFAUlT_LINE_WIDTH;
	iDrawData.iEndPontWidth=DEFAUlT_END_POINT_FACTOR*DEFAUlT_LINE_WIDTH;
	iSpriteFunctions->SizeChangedL();
	iPointStore=new(ELeave) CPointStore();
	iPointStore->ConstructL();
	}

void CHandWritingAnim::Animate(TDateTime* /*aDateTime*/)
	{
	iFunctions->SetInterval(0);
	if (iState==EHwStateWaitingMove)
		SendEatenDownEvent();
	else if (iState==EHwStateWaitingStroke)
		CharacterFinished();
	}

void CHandWritingAnim::Redraw()
	{
	}

void CHandWritingAnim::Command(TInt aOpcode,TAny *aParams)
	{
	switch (aOpcode)
		{
		case EHwOpActivate:
			Activate();
			break;
		case EHwOpDeactivate:
			Deactivate();
			break;
		case EHwOpSetDrawData:;
			SetDrawData(STATIC_CAST(THandwritingDrawData*,aParams));
			break;
		default:
			iFunctions->Panic();
		}
	}

void CHandWritingAnim::FocusChanged(TBool )
	{
	}

TInt CHandWritingAnim::CommandReplyL(TInt aOpcode,TAny *aParams)
	{
	switch (aOpcode)
		{
		case EHwOpSpriteMask:
			SpriteChangeL(*STATIC_CAST(TBool*,aParams));
			break;
		case EHwOpGetLastChar:
			return iLastGeneratedCharacter;
		default:
			iFunctions->Panic();
		}
	return KErrNone;
	}


/*CPointStore*/

CPointStore::CPointStore()
	{}

void CPointStore::ConstructL()
	{
	iPoints=new(ELeave) CArrayFixFlat<TPoint>(16);
	iPoints->ResizeL(256);
	}

void CPointStore::AddPoint(TPoint aPoint)
	{
	if (iNumPoints<256)
		(*iPoints)[iNumPoints++]=aPoint;
	}

TInt CPointStore::GetChar()
	{
	TPoint oldPoint=(*iPoints)[0];
	TPoint newPoint;
	TPoint totalPoint=oldPoint;
	TInt xInc=0,xDec=0,yInc=0,yDec=0;
	TInt yState=0,xState=0;
	TInt ii;
	for (ii=1;ii<iNumPoints;++ii)
		{
		newPoint=(*iPoints)[ii];
		totalPoint+=newPoint;
		if (newPoint.iX>oldPoint.iX)
			++xInc;
		if (newPoint.iX<oldPoint.iX)
			++xDec;
		if (newPoint.iY>oldPoint.iY)
			++yInc;
		if (newPoint.iY<oldPoint.iY)
			++yDec;
		oldPoint=newPoint;
		}
	newPoint-=(*iPoints)[0];
	if (10*yInc<yDec)
		yState=-1;
	else if (yInc>10*yDec)
		yState=1;
	if (10*xInc<xDec)
		xState=-1;
	else if (xInc>10*xDec)
		xState=1;
	if (xState!=0 && yState!=0)
		{
		if (Abs(newPoint.iY)<Abs(newPoint.iX))
			yState=0;
		else
			xState=0;
		}
	if (xState!=0)
		return xState>0 ? EKeyRightArrow:EKeyLeftArrow;
	if (yState!=0)
		return yState>0 ? EKeyDownArrow:EKeyUpArrow;
	TInt firstChar='a';
	TInt numChars=26;
	TInt type=(totalPoint.iY/10)%10;
	if (type>5)
		firstChar='A';
	else if (type==0)
		{
		firstChar='0';
		numChars=10;
		}
	return firstChar+((totalPoint.iX/10)%numChars);
	}

void CPointStore::ClearPoints()
	{
	iNumPoints=0;
	}