textrendering/word/SRC/WPTEST.CPP
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 27 Apr 2010 18:29:58 +0300
branchRCL_3
changeset 19 f7b2d24357ad
parent 0 1fb32624e06b
permissions -rw-r--r--
Revision: 201013 Kit: 201017

/*
* 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: 
* Functions added to Word for test purposes now that Word is no longer a standard EPOC application.
*
*/


#include <e32std.h>
#include <techview/eikcfdlg.h>
#include <techview/eikon.rsg>
#include "WPAPPUI.H"
#include "WPTEXTED.H"
#include "WPTEST.H"
#include "WPTESTPICTURE.H"


struct TKeyAction
	{
	TInt Compare(TInt aKeyCode,TInt aState) const;

	TInt iKeyCode;			// key pressed by user
	TInt iState;			// state in which this action fires
	TInt iCharToInsert;		// character to insert; if 0xFFFF, none
	TInt iNewState;			// new state
	};

TInt TKeyAction::Compare(TInt aKeyCode,TInt aState) const
	{
	if (iKeyCode - aKeyCode)
		return iKeyCode - aKeyCode;
	return iState - aState;
	}

struct TKeyboardMap
	{
	const TKeyAction* Find(TInt aKeyCode,TInt aState) const;

	const TKeyAction* iKeyAction;	// action table
	TInt iKeyActions;				// number of entries in table
	};

const TKeyAction* TKeyboardMap::Find(TInt aKeyCode,TInt aState) const
	{
	const TKeyAction* base = iKeyAction;
	const TKeyAction* end = iKeyAction + iKeyActions;
	while (base < end)
		{
		int n = end - base;
		const TKeyAction* a = &base[n / 2];
		int diff = a->Compare(aKeyCode,aState);
		if (diff > 0)
			end = a;
		else if (diff < 0)
			base = a + 1;
		else
			return a;
		}
	return NULL;
	}

static const TKeyAction TheArabicKeyAction[] =
	{
	{ 'A', 0, 0x0649, 0 },
	{ 'B', 0, 0xFFFF, 0 },
	{ 'C', 0, 0xFFFF, 0 },
	{ 'D', 0, 0x0636, 0 },
	{ 'E', 0, 0xFFFF, 0 },
	{ 'F', 0, 0xFFFF, 0 },
	{ 'G', 0, 0x063A, 0 },
	{ 'H', 0, 0x062D, 0 },
	{ 'I', 0, 0xFFFF, 0 },
	{ 'J', 0, 0xFFFF, 0 },
	{ 'K', 0, 0xFFFF, 0 },
	{ 'L', 0, 0xFFFF, 0 },
	{ 'M', 0, 0xFFFF, 0 },
	{ 'N', 0, 0xFFFF, 0 },
	{ 'O', 0, 0xFFFF, 0 },
	{ 'P', 0, 0xFFFF, 0 },
	{ 'Q', 0, 0xFFFF, 0 },
	{ 'R', 0, 0xFFFF, 0 },
	{ 'S', 0, 0x0635, 0 },
	{ 'T', 0, 0x0637, 0 },
	{ 'U', 0, 0xFFFF, 0 },
	{ 'V', 0, 0xFFFF, 0 },
	{ 'W', 0, 0xFFFF, 0 },
	{ 'X', 0, 0xFFFF, 0 },
	{ 'Y', 0, 0xFFFF, 0 },
	{ 'Z', 0, 0x0638, 0 },
	{ 'a', 0, 0x0627, 0 },
	{ 'b', 0, 0x0628, 0 },
	{ 'c', 0, 0xFFFF, 0 },
	{ 'd', 0, 0x062F, 0 },
	{ 'e', 0, 0xFFFF, 0 },
	{ 'f', 0, 0x0641, 0 },
	{ 'g', 0, 0x0639, 0 },
	{ 'h', 0, 0x0647, 0 },
	{ 'i', 0, 0xFFFF, 0 },
	{ 'j', 0, 0x062C, 0 },
	{ 'k', 0, 0x0643, 0 },
	{ 'l', 0, 0x0644, 0 },
	{ 'm', 0, 0x0645, 0 },
	{ 'n', 0, 0x0646, 0 },
	{ 'o', 0, 0xFFFF, 0 },
	{ 'p', 0, 0x067E, 0 },
	{ 'q', 0, 0x0642, 0 },
	{ 'r', 0, 0x0631, 0 },
	{ 's', 0, 0x0633, 0 },
	{ 't', 0, 0x062A, 0 },
	{ 'u', 0, 0xFFFF, 0 },
	{ 'v', 0, 0xFFFF, 0 },
	{ 'w', 0, 0x0648, 0 },
	{ 'x', 0, 0x062E, 0 },
	{ 'y', 0, 0x064A, 0 },
	{ 'z', 0, 0x0632, 0 },
	};
static const TKeyboardMap TheArabicKeyboardMap =
	{
	TheArabicKeyAction,
	sizeof(TheArabicKeyAction) / sizeof(TKeyAction)
	};

static const TKeyAction TheGreekKeyAction[] =
	{
	{ 'A', 0, 0x0391, 0 },
	{ 'B', 0, 0x0392, 0 },
	{ 'C', 0, 0xFFFF, 0 },
	{ 'D', 0, 0x0394, 0 },
	{ 'E', 0, 0x0395, 0 },
	{ 'F', 0, 0x03A6, 0 },
	{ 'G', 0, 0x0393, 0 },
	{ 'H', 0, 0x03A7, 0 },
	{ 'I', 0, 0x0399, 0 },
	{ 'J', 0, 0x0397, 0 },
	{ 'K', 0, 0x039A, 0 },
	{ 'L', 0, 0x039B, 0 },
	{ 'M', 0, 0x039C, 0 },
	{ 'N', 0, 0x039D, 0 },
	{ 'O', 0, 0x039F, 0 },
	{ 'P', 0, 0x03A0, 0 },
	{ 'Q', 0, 0x0398, 0 },
	{ 'R', 0, 0x03A1, 0 },
	{ 'S', 0, 0x03A3, 0 },
	{ 'T', 0, 0x03A4, 0 },
	{ 'U', 0, 0x03A5, 0 },
	{ 'V', 0, 0x03A8, 0 },
	{ 'W', 0, 0x03A9, 0 },
	{ 'X', 0, 0x039E, 0 },
	{ 'Y', 0, 0xFFFF, 0 },
	{ 'Z', 0, 0x0396, 0 },
	{ 'a', 0, 0x03B1, 0 },
	{ 'b', 0, 0x03B2, 0 },
	{ 'c', 0, 0x03C2, 0 },
	{ 'd', 0, 0x03B4, 0 },
	{ 'e', 0, 0x03B5, 0 },
	{ 'f', 0, 0x03C6, 0 },
	{ 'g', 0, 0x03B3, 0 },
	{ 'h', 0, 0x03C7, 0 },
	{ 'i', 0, 0x03B9, 0 },
	{ 'j', 0, 0x03B7, 0 },
	{ 'k', 0, 0x03BA, 0 },
	{ 'l', 0, 0x03BB, 0 },
	{ 'm', 0, 0x03BC, 0 },
	{ 'n', 0, 0x03BD, 0 },
	{ 'o', 0, 0x03BF, 0 },
	{ 'p', 0, 0x03C0, 0 },
	{ 'q', 0, 0x03B8, 0 },
	{ 'r', 0, 0x03C1, 0 },
	{ 's', 0, 0x03C3, 0 },
	{ 't', 0, 0x03C4, 0 },
	{ 'u', 0, 0x03C5, 0 },
	{ 'v', 0, 0x03C8, 0 },
	{ 'w', 0, 0x03C9, 0 },
	{ 'x', 0, 0x03BE, 0 },
	{ 'y', 0, 0xFFFF, 0 },
	{ 'z', 0, 0x03B6, 0 },
	};
static const TKeyboardMap TheGreekKeyboardMap =
	{
	TheGreekKeyAction,
	sizeof(TheGreekKeyAction) / sizeof(TKeyAction)
	};

static const TKeyAction TheRussianKeyAction[] =
	{
	{ 'A', 0, 0x0410, 0 },
	{ 'B', 0, 0x0411, 0 },
	{ 'C', 0, 0x0426, 0 },
	{ 'D', 0, 0x0414, 0 },
	{ 'E', 0, 0x0415, 0 },
	{ 'F', 0, 0x0424, 0 },
	{ 'G', 0, 0x0413, 0 },
	{ 'H', 0, 0x0425, 0 },
	{ 'I', 0, 0x0418, 0 },
	{ 'J', 0, 0xFFFF, 0 },
	{ 'K', 0, 0x041A, 0 },
	{ 'L', 0, 0x041B, 0 },
	{ 'M', 0, 0x041C, 0 },
	{ 'N', 0, 0x041D, 0 },
	{ 'O', 0, 0x041E, 0 },
	{ 'P', 0, 0x041F, 0 },
	{ 'Q', 0, 0xFFFF, 0 },
	{ 'R', 0, 0x0420, 0 },
	{ 'S', 0, 0x0421, 0 },
	{ 'T', 0, 0x0422, 0 },
	{ 'U', 0, 0x0423, 0 },
	{ 'V', 0, 0x0412, 0 },
	{ 'W', 0, 0xFFFF, 0 },
	{ 'X', 0, 0xFFFF, 0 },
	{ 'Y', 0, 0x0419, 0 },
	{ 'Z', 0, 0x0417, 0 },
	{ 'a', 0, 0x0430, 0 },
	{ 'b', 0, 0x0431, 0 },
	{ 'c', 0, 0x0446, 0 },
	{ 'd', 0, 0x0434, 0 },
	{ 'e', 0, 0x0435, 0 },
	{ 'f', 0, 0x0444, 0 },
	{ 'g', 0, 0x0433, 0 },
	{ 'h', 0, 0x0445, 0 },
	{ 'i', 0, 0x0438, 0 },
	{ 'j', 0, 0xFFFF, 0 },
	{ 'k', 0, 0x043A, 0 },
	{ 'l', 0, 0x043B, 0 },
	{ 'm', 0, 0x043C, 0 },
	{ 'n', 0, 0x043D, 0 },
	{ 'o', 0, 0x043E, 0 },
	{ 'p', 0, 0x043F, 0 },
	{ 'q', 0, 0xFFFF, 0 },
	{ 'r', 0, 0x0440, 0 },
	{ 's', 0, 0x0441, 0 },
	{ 't', 0, 0x0442, 0 },
	{ 'u', 0, 0x0443, 0 },
	{ 'v', 0, 0x0432, 0 },
	{ 'w', 0, 0xFFFF, 0 },
	{ 'x', 0, 0xFFFF, 0 },
	{ 'y', 0, 0x0439, 0 },
	{ 'z', 0, 0x0437, 0 },
	};
static const TKeyboardMap TheRussianKeyboardMap =
	{
	TheRussianKeyAction,
	sizeof(TheRussianKeyAction) / sizeof(TKeyAction)
	};

CWordTest* CWordTest::NewL(CTextView* aTextView)
	{
	return new(ELeave) CWordTest(aTextView);
	}

CWordTest::CWordTest(CTextView* aTextView):
	iTextView(aTextView),
	iKeyboard(EStandardKeyboard),
	iKeyboardState(0),
	iRuledPaper(FALSE),
	iBackgroundBitmap(NULL),
	iTruncateWithEllipsis(FALSE),
	iCursorFlashing(TRUE),
	iCursorPlacement(ECursorVertical),
	iCursorWeight(3),
	iCursorXorColor(KRgbWhite),
	iStyleIndex(0),
	iPositioningHint(TCursorPosition::EPosHintUndefined)
	{
	}

CWordTest::~CWordTest()
	{
	delete iBackgroundBitmap;
	}

TKeyResponse CWordTest::OfferKeyEventL(const TKeyEvent& aKeyEvent,TEventCode aEventCode,TChar& aCharToInsert)
	{
	if (iKeyboard == EStandardKeyboard)
		return EKeyWasNotConsumed;
	if (aEventCode != EEventKey)
		return EKeyWasNotConsumed;
	if (aKeyEvent.iCode >= ESpecialKeyBase && aKeyEvent.iCode < ESpecialKeyBase + ESpecialKeyCount)
		return EKeyWasNotConsumed;
	if (aKeyEvent.iCode >= ENonCharacterKeyBase && aKeyEvent.iCode < ENonCharacterKeyBase + ENonCharacterKeyCount)
		return EKeyWasNotConsumed;
	aCharToInsert = 0xFFFF;
		
	const TKeyboardMap* map = NULL;
	switch (iKeyboard)
		{
	case EArabicKeyboard:
		map = &TheArabicKeyboardMap;
		break;
	case EGreekKeyboard:
		map = &TheGreekKeyboardMap;
		break;
	case ERussianKeyboard:
		map = &TheRussianKeyboardMap;
		break;
	default:
		break;
		}
	if (map)
		{
		const TKeyAction* action = map->Find(aKeyEvent.iCode,iKeyboardState);
		if (action)
			{
			aCharToInsert = action->iCharToInsert;
			iKeyboardState = action->iNewState;
			return EKeyWasConsumed;
			}
		}
	return EKeyWasNotConsumed;
	}

void CWordTest::SetKeyboardL(TKeyboardCode aKeyboard)
	{
	switch (aKeyboard)
		{
	case EArabicKeyboard:
		SetCursorPositioningHintL(TCursorPosition::EInsertStrongR2L);
		break;
	default:
		SetCursorPositioningHintL(TCursorPosition::EInsertStrongL2R);
		break;
		}
	iKeyboard = aKeyboard;
	}

void CWordTest::SetPictureAlignmentL(CEikRichTextEditor* aEditor,TFontPresentation::TAlignment aAlignment)
	{
	TCharFormat format;
	format.iFontPresentation.iPictureAlignment = aAlignment;
	TCharFormatMask mask;
	mask.SetAttrib(EAttFontPictureAlignment);
	aEditor->ApplyCharFormatL(format,mask);
	}

void CWordTest::SetWrapL(CEikRichTextEditor* aEditor,TBool aWrap)
	{
	CParaFormat format;
	format.iWrap = aWrap;
	TParaFormatMask mask;
	mask.SetAttrib(EAttWrap);
	aEditor->ApplyParaFormatL(&format,mask);
	}
	
void CWordTest::InsertPictureL(CEikRichTextEditor* aEditor)
	{
	CWordTestPicture* pic=CWordTestPicture::NewLC();
	TPictureHeader header;
   	header.iPictureType = KUidWordTestPictureType;
   	header.iPicture=pic;
 		
 	TInt pos = aEditor->CursorPos();
 	aEditor->RichText()->InsertL(pos, header); 
//CRichText::InsertL takes ownership of the picture via TPictureHeader 	
 	CleanupStack::Pop(pic); 
 	
 	aEditor->HandleTextChangedL();
 	aEditor->SetCursorPosL(pos+1, EFalse);	
	}

void CWordTest::ToggleCaseL(CEikRichTextEditor* aEditor)
	{
	TCursorSelection sel = aEditor->Selection();
	int start = sel.LowerPos();
	int end = start;
	TBool to_upper = TRUE;
	while (end < sel.HigherPos())
		{
		TPtrC text;
		TCharFormat f;
		aEditor->RichText()->GetChars(text,f,start);
		int length = text.Length();
		if (start + length > sel.HigherPos())
			length = sel.HigherPos() - start;
		end = start + length;
		if (start == sel.LowerPos() && end > start)
			{
			if (TChar(text[0]).IsUpper())
				to_upper = FALSE;
			}
		TText* p = (TText*)text.Ptr();
		TText* q = p + length;
		while (p < q)
			{
			if (to_upper)
				*p = (TText)(TChar(*p).GetUpperCase());
			else
				*p = (TText)(TChar(*p).GetLowerCase());
			p++;
			}

		start = end;
		}
	aEditor->TextView()->HandleRangeFormatChangeL(sel);
	}

void CWordTest::BenchmarkL()
	{
	TTime start, end;
	start.HomeTime();
	const int n = 100;
	for (int i = 0; i < n; i++)
		iTextView->FormatTextL();
	end.HomeTime();
	int ms = (I64LOW(end.Int64()) - I64LOW(start.Int64())) / 1000;
	TBuf<128> message;
#ifdef _DEBUG
	_LIT(build,"debug");
#else
	_LIT(build,"release");
#endif
	message.Format(_L("form%d %S: reformatting %d times took %d milliseconds"),
		iTextView->Layout()->MajorVersion(),&build,n,ms);
	User::InfoPrint(message);
	}

void CWordTest::SetTruncateWithEllipsisL(CEikRichTextEditor* aEditor,TBool aOn)
	{
	iTruncateWithEllipsis = aOn;
	aEditor->TextLayout()->SetTruncating(aOn);
	aEditor->NotifyNewFormatL();
	}

void CWordTest::ToggleCursorFlash()
	{
	iCursorFlashing = !iCursorFlashing;
	iTextView->SetCursorFlash(iCursorFlashing);
	}

void CWordTest::SetCursorPlacement(TTmCursorPlacement aPlacement)
	{
	iCursorPlacement = aPlacement;
	iTextView->SetCursorPlacement(aPlacement);
	}

void CWordTest::SetCursorWeight(TInt aWeight)
	{
	iCursorWeight = aWeight;
	iTextView->SetCursorWeight(aWeight);
	}

void CWordTest::SetCursorXorColor(TRgb aColor)
	{
	iCursorXorColor = aColor;
	iTextView->SetCursorXorColor(aColor);
	}

void CWordTest::TurnOnCustomDrawing(CTextLayout* aLayout)
	{
	if (aLayout->MajorVersion() == 2) // custom drawing is not supported in FORM1
		aLayout->SetCustomDraw(this);
	}

void CWordTest::SetSelectionHighlightStyle(TInt aStyleIndex)
	{
	iStyleIndex = aStyleIndex;
	}

TInt CWordTest::SelectionHighlightStyle() const
	{
	return iStyleIndex;
	}

void CWordTest::DrawLineGraphics(const TParam& aParam,const TLineInfo& aLineInfo) const
	{
	if (iRuledPaper)
		{
		aParam.iGc.SetPenColor(KRgbRed);
		TPoint p(aLineInfo.iOuterRect.iTl.iX,aLineInfo.iInnerRect.iBr.iY - 1);
		TPoint q(aLineInfo.iOuterRect.iBr.iX,aLineInfo.iInnerRect.iBr.iY - 1);
		aParam.iGc.DrawLine(p,q);
		}
	}

void CWordTest::DrawText(const TParam& aParam,const TLineInfo& aLineInfo,const TCharFormat& aFormat,const TDesC& aText,
						 const TPoint& aTextOrigin,TInt aExtraPixels) const
	{
	/*
	Draw outlined text by drawing the text nine times: eight times in the background colour, moving the origin
	one pixel up, down, right, and left to make a box, and once in the foreground colour, not offset.
	*/
	if (aFormat.iFontPresentation.iHighlightStyle == TFontPresentation::EFontHighlightFirstCustomStyle)
		{
		aParam.iGc.SetPenColor(aFormat.iFontPresentation.iTextColor);
		for (int x = -1; x <= 1; x++)
			for (int y = -1; y <= 1; y++)
				if (x || y)
					{
					if (aExtraPixels)
						aParam.iGc.SetCharJustification(aExtraPixels,aText.Length());
					TPoint p(aTextOrigin);
					p.iX += x;
					p.iY += y;
					aParam.iGc.DrawText(aText,p);
					}
		aParam.iGc.SetPenColor(aFormat.iFontPresentation.iHighlightColor);
		}
	MFormCustomDraw::DrawText(aParam,aLineInfo,aFormat,aText,aTextOrigin,aExtraPixels);
	}

void CWordTest::DrawBackground(const TParam& aParam,const TRgb& aBackground,TRect& aDrawn) const
	{
	if (iBackgroundBitmap)
		{
		aParam.iGc.SetClippingRect(aParam.iDrawRect);
		aDrawn.SetRect(aParam.iTextLayoutTopLeft,iBackgroundBitmap->SizeInPixels());
		aParam.iGc.DrawBitmap(aDrawn,iBackgroundBitmap);
		}
	else
		MFormCustomDraw::DrawBackground(aParam,aBackground,aDrawn);
	}

TRgb CWordTest::SystemColor(TUint aColorIndex, TRgb aDefaultColor) const
	{
	if (aColorIndex == TLogicalRgb::ESystemSelectionForegroundIndex)
		{
		switch (iStyleIndex)
			{
		case 1:
			return KRgbBlue;
		case 2:
			return KRgbWhite;
		default:
			break;
			}
		}
	else if (aColorIndex == TLogicalRgb::ESystemSelectionBackgroundIndex)
		{
		switch (iStyleIndex)
			{
		case 1:
			return KRgbBlue;
		case 2:
			return KRgbRed;
		default:
			break;
			}
		}
	return aDefaultColor;
	}

void CWordTest::SetRuledPaperL(CEikRichTextEditor* aEditor,TBool aOn)
	{
	iRuledPaper = aOn;
	aEditor->NotifyNewFormatL();
	}

void CWordTest::SetBackgroundBitmapL(CEikRichTextEditor* aEditor,TBool aOn)
	{
	if (iBackgroundBitmap)
		delete iBackgroundBitmap;
	iBackgroundBitmap = NULL;
	if (aOn)
		{
		_LIT(KPathMask, "_:\\");
		TFileName filename(KPathMask);
		filename[0] = 'A' + static_cast<TInt>(RFs::GetSystemDrive());
		
		CEikFileOpenDialog* dialog = new(ELeave) CEikFileOpenDialog(&filename);
		if (dialog->ExecuteLD(R_EIK_DIALOG_FILE_OPEN))
			{
			iBackgroundBitmap = new(ELeave) CFbsBitmap;
			if (iBackgroundBitmap->Load(filename))
				User::InfoPrint(_L("cannot load this file as a bitmap"));
			}
		}
	aEditor->NotifyNewFormatL();
	}

void CWordTest::SetCursorPositioningHintL(
	TCursorPosition::TPosHint aHint)
	{
	iTextView->SetCursorPositioningHintL(aHint);
	iPositioningHint = aHint;
	}

TCursorPosition::TPosHint CWordTest::CursorPositioningHint() const
	{
	return iPositioningHint;
	}