os/graphics/graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgditext.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "swdirectgdiengine.h"
    17 #include <bitdrawinterfaceid.h>
    18 #include <bmalphablend.h>
    19 
    20 /**
    21 @see MDirectGdiEngine::SetFont()
    22 */
    23 void CSwDirectGdiEngine::SetFont(TUint32 /*aFontId*/)
    24 	{
    25 	// Do nothing.
    26 	// The SW adapter doesn't need the font ID to index a separate font cache because it uses the standard one.
    27 	}
    28 
    29 
    30 /**
    31 @see MDirectGdiEngine::ResetFont()
    32 */
    33 void CSwDirectGdiEngine::ResetFont()
    34 	{
    35 	}
    36 
    37 
    38 /**
    39 @see MDirectGdiEngine::SetTextShadowColor()
    40 */
    41 void CSwDirectGdiEngine::SetTextShadowColor(const TRgb& aColor)
    42 	{	
    43 	iTextShadowColor = aColor;
    44 	}
    45 
    46 /**
    47 @see MDirectGdiEngine::BeginDrawGlyph()
    48 */
    49 void CSwDirectGdiEngine::BeginDrawGlyph()
    50 	{
    51 	}
    52 
    53 /**
    54 @see MDirectGdiEngine::DrawGlyph()
    55 @panic DGDIAdapter 56, if an invalid glyph bitmap type is passed in.
    56 */
    57 void CSwDirectGdiEngine::DrawGlyph(const TPoint& aScrPos, const TChar /*aChar*/, const TUint8* aGlyphImage,
    58 								   const TGlyphBitmapType aGlyphBitmapType, const TSize& aGlyphImageSize,
    59 								   const TRect& aScrClipRect, const DirectGdi::TGraphicsRotation aRotation)
    60 	{
    61 	TPoint pos=aScrPos;
    62 	pos+=iDrawOrigin;
    63 	TRect clipRect=aScrClipRect;
    64 	clipRect.iTl+=iDrawOrigin;
    65 	clipRect.iBr+=iDrawOrigin;
    66 	TRect regionRect(0, 0, 0, 0);
    67 	TInt numRects = iDefaultRegionPtr->Count();
    68 	for (TInt count = 0; count < numRects; count++)
    69 		{
    70 		// Do the clip rects intersect?
    71 		regionRect = (*iDefaultRegionPtr)[count];
    72 		if (!regionRect.Intersects(clipRect))
    73 			{
    74 			// Nothing to draw
    75 			continue;
    76 			}
    77 		// Clip to intersection of two clip rects
    78 		regionRect.Intersection(clipRect);
    79 		
    80 		if (aRotation == DirectGdi::EGraphicsRotationNone)	// Horizontal text
    81 			{
    82 			// Do the glyph and the clip rect intersect?
    83 			TRect glyphRect(pos, aGlyphImageSize);
    84 			if (!regionRect.Intersects(glyphRect))
    85 				{
    86 				// Nothing to draw
    87 				continue;
    88 				}
    89 			// Clip to intersection with glyph bitmap 
    90 			regionRect.Intersection(glyphRect);
    91 			
    92 			switch (aGlyphBitmapType)
    93 				{
    94 				case EMonochromeGlyphBitmap:
    95 					{
    96 					DrawBitmapGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
    97 					break;
    98 					}
    99 				case EAntiAliasedGlyphBitmap:
   100 					{
   101 					DrawAntiAliasedGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
   102 					break;
   103 					}
   104 				case EFourColourBlendGlyphBitmap:
   105 					{
   106 					DrawFourColourGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
   107 					break;
   108 					}
   109 				default:
   110 					GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidGlyphBitmapType);
   111 				}
   112 			}
   113 		else	// Vertical text
   114 			{
   115 			/*
   116 			// Do the glyph and the clip rect intersect?
   117 			TRect glyphRect(aPos, aGlyphImageSize);
   118 			if (!regionRect.Intersects(glyphRect))
   119 				{
   120 				// Nothing to draw
   121 				continue;
   122 				}
   123 			// Clip to intersection with glyph bitmap 
   124 			regionRect.Intersection(glyphRect);
   125 			*/
   126 			
   127 			switch (aGlyphBitmapType)
   128 				{
   129 				case EMonochromeGlyphBitmap:
   130 					{
   131 					DrawRotatedBitmapGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
   132 					break;
   133 					}
   134 				case EAntiAliasedGlyphBitmap:
   135 					{
   136 					DrawRotatedAntiAliasedGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
   137 					break;
   138 					}
   139 				case EFourColourBlendGlyphBitmap:
   140 					{
   141 					DrawRotatedFourColourGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
   142 					break;
   143 					}
   144 				default:
   145 					GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidGlyphBitmapType);
   146 				}
   147 			}
   148 		// Now display it
   149 		iDrawDevice->UpdateRegion(regionRect);
   150 		}
   151 	}
   152 
   153 /**
   154 @see MDirectGdiEngine::EndDrawGlyph()
   155 */
   156 void CSwDirectGdiEngine::EndDrawGlyph()
   157 	{
   158 	}
   159 
   160 /**
   161 Draw a bitmap glyph.
   162 	
   163 @param	aPos			Position to start drawing gyph.
   164 @param	aGlyphImage		Pointer to the glyph image data.
   165 @param	aGlyphImageSize	Glyph image size.
   166 @param	aClipRect		Clipping rect.
   167 */
   168 void CSwDirectGdiEngine::DrawBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   169 										 const TSize& aGlyphImageSize, const TRect& aClipRect)
   170 	{
   171 	// aChar parameter not needed because SW implementation uses the default glyph cache
   172 	// therefore does not need aChar to index its own local cache
   173 	
   174 	/*
   175 	Divert if the character is large.
   176 	Large is defined as wider than 30 bits (a scan line won't fit in a TInt32)
   177 	or greater than 32 bits high (because that's the current array size - could be changed).
   178 	*/
   179 	TInt dataHeight = aGlyphImageSize.iHeight;
   180 	TInt dataLength = aGlyphImageSize.iWidth;
   181 	if (dataLength > 30 || dataHeight > 32)
   182 		{
   183 		DrawLargeBitmapGlyph(aPos, aGlyphImage, aGlyphImageSize, aClipRect);
   184 		return;
   185 		}
   186 	
   187 	TInt bitIndex = 0;
   188 	TInt16 repeatCount = 0;
   189 	TUint32 binaryData[32];
   190 	TUint32* binaryDataPtr = binaryData;
   191 	TUint32* binaryDataPtrLimit;
   192 	for (TInt glyphLine = 0; glyphLine < dataHeight; glyphLine += repeatCount) // for lines in the glyph bitmap
   193 		{
   194 		repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
   195 		repeatCount >>= bitIndex & 7;
   196 		TInt multiLineFlag = repeatCount & 1;
   197 		repeatCount >>= 1;
   198 		repeatCount &= 0xf;
   199 		bitIndex += 5;
   200 		binaryDataPtrLimit = binaryData + glyphLine + repeatCount;
   201 		if (multiLineFlag)
   202 			{
   203 			while (binaryDataPtr < binaryDataPtrLimit)
   204 				{
   205 				TInt glyphDataOffsetPtr = TInt(aGlyphImage) + (bitIndex >> 3);
   206 				TUint32* glyphDataWord = (TUint32*)(glyphDataOffsetPtr & ~3);
   207 				TInt bitShift = bitIndex & 7;
   208 				bitShift += (glyphDataOffsetPtr & 3) << 3;
   209 				*binaryDataPtr = (*glyphDataWord++) >> bitShift;
   210 				if (bitShift)
   211 					{
   212 					*binaryDataPtr |= (*glyphDataWord << (32 - bitShift));
   213 					}
   214 				bitIndex += dataLength;
   215 				binaryDataPtr++;
   216 				}
   217 			}
   218 		else
   219 			{
   220 			TInt glyphDataOffsetPtr = TInt(aGlyphImage) + (bitIndex >> 3);
   221 			TUint32* glyphDataWord = (TUint32*)(glyphDataOffsetPtr & ~3);
   222 			TInt bitShift = bitIndex & 7;
   223 			bitShift += (glyphDataOffsetPtr & 3) << 3;
   224 			TUint32 data = (*glyphDataWord++) >> bitShift;
   225 			if (bitShift)
   226 				{
   227 				data |= (*glyphDataWord << (32 - bitShift));
   228 				}
   229 			while (binaryDataPtr < binaryDataPtrLimit)
   230 				{
   231 				*binaryDataPtr++ = data;
   232 				}
   233 			bitIndex += dataLength;
   234 			}
   235 		}
   236 	TPoint topLeft(aPos);
   237 	binaryDataPtr = ClipBinaryArray(binaryData, binaryData + dataHeight, dataLength, dataHeight, topLeft, aClipRect);
   238 	if ((dataLength > 0) && (dataHeight > 0))
   239 		{
   240 		iDrawDevice->WriteBinary(topLeft.iX, topLeft.iY, binaryDataPtr, dataLength, dataHeight, iPenColor, GcDrawMode(iDrawMode) );
   241 		}
   242 	}
   243 	
   244 	
   245 /**
   246 Draw a large bitmap glyph.
   247 	
   248 @param	aPos			Position to start drawing gyph.
   249 @param	aGlyphImage		Pointer to the glyph image data.
   250 @param	aGlyphImageSize	Glyph image size.
   251 @param	aClipRect		Clipping rect.
   252 */
   253 void CSwDirectGdiEngine::DrawLargeBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   254 											 const TSize& aGlyphImageSize, const TRect& aClipRect)
   255 	{
   256 	TPoint printPos(aPos);
   257 	const TInt dataLength = aGlyphImageSize.iWidth;
   258 	const TInt dataHeight = aGlyphImageSize.iHeight;
   259 	TInt bitIndex = 0;
   260 	TInt16 repeatCount = 0;
   261 	TUint32* scanLineBuffer = iDrawDevice->ScanLineBuffer();
   262 	const TInt scanLineWords = (iDrawDevice->ScanLineBytes()) << 3;
   263 	for (TInt glyphLine = 0; glyphLine < dataHeight; glyphLine += repeatCount) // for lines in the glyph bitmap
   264 		{
   265 		repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
   266 		repeatCount >>= bitIndex & 7;
   267 		const TInt multiLineFlag = repeatCount & 1;
   268 		repeatCount >>= 1;
   269 		repeatCount &= 0xf;
   270 		bitIndex += 5;
   271 		if (multiLineFlag)
   272 			{
   273 			for (TInt currentLine = 0; currentLine < repeatCount; currentLine++)
   274 				{
   275 				CopyCharLine(scanLineBuffer, scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
   276 				OutputCharLineMultiplied(printPos, scanLineBuffer, dataLength, 1, aClipRect);
   277 				bitIndex += dataLength;
   278 				printPos.iY++;
   279 				}
   280 			}
   281 		else
   282 			{
   283 			CopyCharLine(scanLineBuffer, scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
   284 			OutputCharLineMultiplied(printPos, scanLineBuffer, dataLength, repeatCount, aClipRect);
   285 			printPos.iY += repeatCount;
   286 			bitIndex += dataLength;
   287 			}
   288 		}
   289 	}
   290 
   291 
   292 /**
   293 
   294 */
   295 void CSwDirectGdiEngine::CopyCharLine(TUint32* aBinaryDataPtr, TInt aBufferWords, const TUint8* aData, TInt aBitShift, TInt aCharWidth)
   296 	{
   297 	aBitShift &= 7;
   298 	TInt wordsToCopy = (aCharWidth + 31) >> 5;
   299 	if (wordsToCopy > aBufferWords)
   300 		{
   301 		wordsToCopy = aBufferWords;
   302 		}
   303 	TUint32* ptrLimit = aBinaryDataPtr + wordsToCopy;
   304 	TUint32* dataWord = (TUint32*)(TInt(aData) & ~3);
   305 	aBitShift += (TInt(aData) - TInt(dataWord)) << 3;
   306 	while (aBinaryDataPtr < ptrLimit)
   307 		{
   308 		*aBinaryDataPtr = *dataWord++;
   309 		*aBinaryDataPtr >>= aBitShift;
   310 		if (aBitShift)
   311 			{
   312 			*aBinaryDataPtr |= (*dataWord << (32 - aBitShift));
   313 			}
   314 		aBinaryDataPtr++;
   315 		}
   316 	}
   317 
   318 
   319 /**
   320 
   321 */
   322 void CSwDirectGdiEngine::OutputCharLineMultiplied(TPoint aPrintPos, TUint32* aBuffer, TInt aDataLength, TInt aNum, const TRect& aClipRect)
   323 	{
   324 	if (aDataLength <= 0)
   325 		{
   326 		return;
   327 		}
   328 	TInt bufferWords = (aDataLength + 31) >> 5;
   329 	TUint32* bufferLimit = aBuffer + bufferWords;
   330 	if (aPrintPos.iX < aClipRect.iTl.iX)
   331 		{
   332 		TInt pixelExcess = aClipRect.iTl.iX - aPrintPos.iX;
   333 		while (pixelExcess >= 32)
   334 			{
   335 			aBuffer++;
   336 			bufferWords--;
   337 			aDataLength -= 32;
   338 			pixelExcess -= 32;
   339 			}
   340 		if (aDataLength <= 0)
   341 			{
   342 			return;
   343 			}
   344 		if (pixelExcess > 0)
   345 			{
   346 			TInt shiftUp = 32 - pixelExcess;
   347 			TUint32* bufferPtr = aBuffer;
   348 			while (bufferPtr < bufferLimit)
   349 				{
   350 				*bufferPtr >>= pixelExcess;
   351 				if (bufferPtr < bufferLimit - 1)
   352 					{
   353 					*bufferPtr |= (*(bufferPtr + 1) << shiftUp);
   354 					}
   355 				bufferPtr++;
   356 				}
   357 			aDataLength -= pixelExcess;
   358 			if (aDataLength <= 0)
   359 				{
   360 				return;
   361 				}
   362 			}
   363 		aPrintPos.iX = aClipRect.iTl.iX;
   364 		}
   365 	if (aPrintPos.iX + aDataLength > aClipRect.iBr.iX)
   366 		{
   367 		TInt pixelExcess = aPrintPos.iX + aDataLength - aClipRect.iBr.iX;
   368 		aDataLength -= pixelExcess;
   369 		if (aDataLength <= 0)
   370 			{
   371 			return;
   372 			}
   373 		}
   374 	while (aNum > 0)
   375 		{
   376 		if ((aPrintPos.iY >= aClipRect.iTl.iY) && (aPrintPos.iY < aClipRect.iBr.iY))
   377 			{
   378 			iDrawDevice->WriteBinaryLine(aPrintPos.iX, aPrintPos.iY, aBuffer, aDataLength, iPenColor, GcDrawMode(iDrawMode));
   379 			}
   380 		aPrintPos.iY++;
   381 		aNum--;
   382 		}
   383 	}
   384 
   385 
   386 /**
   387 Draw a rotated bitmap glyph.
   388 	
   389 @param	aPos			Position to start drawing glyph after rotation has been applied.
   390 @param	aGlyphImage		Pointer to the glyph image data.
   391 @param	aGlyphImageSize	Glyph image size.
   392 @param	aClipRect		Clipping rect.
   393 @param	aRotation		Rotation specifying how the glyph will be drawn.
   394 */
   395 void CSwDirectGdiEngine::DrawRotatedBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   396 												const TSize& aGlyphImageSize, const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
   397 	{
   398 	TPoint printPos(aPos);
   399 	TInt dataLength = aGlyphImageSize.iWidth;
   400 	TInt dataHeight = aGlyphImageSize.iHeight;
   401 	TInt bitIndex = 0;
   402 	TInt16 repeatCount = 0;
   403 	TInt direction = (aRotation == DirectGdi::EGraphicsRotation270) ? 1 : -1;
   404 	TUint32* scanLineBuffer = iDrawDevice->ScanLineBuffer();
   405 	TInt scanLineWords = (iDrawDevice->ScanLineBytes()) << 3;
   406 	for (TInt glyphLine = 0; glyphLine < dataHeight; glyphLine += repeatCount) // for lines in the glyph bitmap...
   407 		{
   408 		repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
   409 		repeatCount >>= bitIndex & 7;
   410 		TInt multiLineFlag = repeatCount & 1;
   411 		repeatCount >>= 1;
   412 		repeatCount &= 0xf;
   413 		bitIndex += 5;
   414 		TInt signedRepeatCount = repeatCount * direction;
   415 		if (multiLineFlag)
   416 			{
   417 			for (TInt currentLine = 0; currentLine < repeatCount; currentLine++)
   418 				{
   419 				CopyCharLine(scanLineBuffer, scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
   420 				OutputCharLineVertical(printPos, scanLineBuffer, dataLength, 1, direction, aClipRect);
   421 				bitIndex += dataLength;
   422 				printPos.iX += direction;
   423 				}
   424 			}
   425 		else
   426 			{
   427 			CopyCharLine(scanLineBuffer,scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
   428 			OutputCharLineVertical(printPos, scanLineBuffer, dataLength, repeatCount, direction, aClipRect);
   429 			printPos.iX += signedRepeatCount;
   430 			bitIndex += dataLength;
   431 			}
   432 		}
   433 	}
   434 
   435 
   436 /**
   437 
   438 */
   439 void CSwDirectGdiEngine::OutputCharLineVertical(TPoint aPrintPos, TUint32* aBuffer, TInt aDataLength,
   440 												TInt aNum, TInt aDirection, const TRect& aClipRect)
   441 	{
   442 //	const DirectGdi::TGraphicsRotation rotation = aUp ? DirectGdi::EGraphicsRotation270 : DirectGdi::EGraphicsRotation90;
   443 //	TInt direction = (aRotation == DirectGdi::EGraphicsRotation270) ? 1 : -1;
   444 	if (aDataLength <= 0)
   445 		{
   446 		return;
   447 		}
   448 	TInt bufferWords = (aDataLength + 31) >> 5;
   449 	TUint32* bufferLimit = aBuffer + bufferWords;
   450 	if (aDirection == 1)
   451 		{
   452 		if (aPrintPos.iY >= aClipRect.iBr.iY)
   453 			{
   454 			TInt pixelExcess = aPrintPos.iY - aClipRect.iBr.iY + 1;
   455 			while (pixelExcess >= 32)
   456 				{
   457 				aBuffer++;
   458 				aDataLength -= 32;
   459 				pixelExcess -= 32;
   460 				}
   461 			if (aDataLength <= 0)
   462 				{
   463 				return;
   464 				}
   465 			if (pixelExcess > 0)
   466 				{
   467 				TInt shiftUp = 32 - pixelExcess;
   468 				TUint32* bufferPtr = aBuffer;
   469 				while (bufferPtr < bufferLimit)
   470 					{
   471 					*bufferPtr >>= pixelExcess;
   472 					if (bufferPtr < bufferLimit - 1)
   473 						*bufferPtr |= (*(bufferPtr + 1) << shiftUp);
   474 					bufferPtr++;
   475 					}
   476 				aDataLength -= pixelExcess;
   477 				if (aDataLength <= 0)
   478 					{
   479 					return;
   480 					}
   481 				}
   482 			aPrintPos.iY = aClipRect.iBr.iY - 1;
   483 			}
   484 		if ((aPrintPos.iY - aDataLength) < (aClipRect.iTl.iY - 1))
   485 			{
   486 			TInt pixelExcess = aClipRect.iTl.iY - 1 - aPrintPos.iY + aDataLength;
   487 			aDataLength -= pixelExcess;
   488 			if (aDataLength <= 0)
   489 				{
   490 				return;
   491 				}
   492 			}
   493 		}
   494 	else
   495 		{
   496 		if (aPrintPos.iY < aClipRect.iTl.iY)
   497 			{
   498 			TInt pixelExcess = aClipRect.iTl.iY - aPrintPos.iY;
   499 			while (pixelExcess >= 32)
   500 				{
   501 				aBuffer++;
   502 				aDataLength -= 32;
   503 				pixelExcess -= 32;
   504 				}
   505 			if (aDataLength <= 0)
   506 				{
   507 				return;
   508 				}
   509 			if (pixelExcess > 0)
   510 				{
   511 				TInt shiftup = 32 - pixelExcess;
   512 				TUint32* bufferptr = aBuffer;
   513 				while (bufferptr < bufferLimit)
   514 					{
   515 					*bufferptr >>= pixelExcess;
   516 					if (bufferptr < bufferLimit - 1)
   517 						*bufferptr |= (*(bufferptr + 1) << shiftup);
   518 					bufferptr++;
   519 					}
   520 				aDataLength -= pixelExcess;
   521 				if (aDataLength <= 0)
   522 					{
   523 					return;
   524 					}
   525 				}
   526 			aPrintPos.iY = aClipRect.iTl.iY;
   527 			}
   528 		if (aPrintPos.iY + aDataLength > aClipRect.iBr.iY)
   529 			{
   530 			TInt pixelExcess = aPrintPos.iY + aDataLength - aClipRect.iBr.iY;
   531 			aDataLength -= pixelExcess;
   532 			if (aDataLength <= 0)
   533 				{
   534 				return;
   535 				}
   536 			}
   537 		}
   538 	CGraphicsContext::TDrawMode drawMode = GcDrawMode(iDrawMode);
   539 	while (aNum > 0)
   540 		{
   541 		if ((aPrintPos.iX >= aClipRect.iTl.iX) && (aPrintPos.iX < aClipRect.iBr.iX))
   542 			iDrawDevice->WriteBinaryLineVertical(aPrintPos.iX, aPrintPos.iY, aBuffer, aDataLength, iPenColor, drawMode, (aDirection == 1));
   543 		aPrintPos.iX += aDirection;
   544 		aNum--;
   545 		}
   546 	}
   547 
   548 
   549 /**
   550 Draw an antialiased glyph.
   551 	
   552 @param	aPos			Position to start drawing gyph.
   553 @param	aGlyphImage		Pointer to the glyph image data.
   554 @param	aGlyphImageSize	Glyph image size.
   555 @param	aClipRect		Clipping rect.
   556 */
   557 void CSwDirectGdiEngine::DrawAntiAliasedGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   558 											  const TSize& aGlyphImageSize, const TRect& aClipRect)
   559 	{
   560 	const TInt topRow = Max(0, aClipRect.iTl.iY - aPos.iY);
   561 	const TInt bottomRow = Min(aGlyphImageSize.iHeight, aClipRect.iBr.iY - aPos.iY);
   562 	const TInt leftCol = Max(0, aClipRect.iTl.iX - aPos.iX);
   563 	const TInt rightCol = Min(aGlyphImageSize.iWidth, aClipRect.iBr.iX - aPos.iX);
   564 	const TUint8* p = aGlyphImage + (topRow * aGlyphImageSize.iWidth) + leftCol;
   565 	const TInt x = aPos.iX + leftCol;
   566 	TInt y = aPos.iY + topRow;
   567 	const TInt cols = rightCol - leftCol;
   568 
   569 	for (TInt row = topRow; row < bottomRow; row++, p += aGlyphImageSize.iWidth, y++)
   570 		{
   571 		iDrawDevice->WriteRgbAlphaMulti(x, y, cols, iPenColor, p);
   572 		}
   573 	}
   574 
   575 
   576 /**
   577 Draw a rotated antialiased glyph.
   578 	
   579 @param	aPos			Position to start drawing gyph after rotation has been applied.
   580 @param	aGlyphImage		Pointer to the glyph image data.
   581 @param	aGlyphImageSize	Glyph image size.
   582 @param	aClipRect		Clipping rect.
   583 @param	aRotation		Rotation specifying how the glyph will be drawn.
   584 */
   585 void CSwDirectGdiEngine::DrawRotatedAntiAliasedGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   586 													 const TSize& aGlyphImageSize, const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
   587 	{
   588 	const int KBufferSize = 32;
   589 	TUint8 maskBuffer[KBufferSize];
   590 	int topRow = 0;
   591 	int bottomRow = 0;
   592 	int leftCol = 0;
   593 	int rightCol = 0;
   594 	const TUint32 penColor = iPenColor.Internal();
   595 	const TUint32 brushColor = iBrushColor.Internal();
   596 	
   597 	if (aRotation == DirectGdi::EGraphicsRotation270)
   598 		{
   599 		topRow = Max(0, aClipRect.iTl.iX - aPos.iX);
   600 		bottomRow = Min(aGlyphImageSize.iHeight, aClipRect.iBr.iX - aPos.iX);
   601 		leftCol = Max(0, aPos.iY - aClipRect.iBr.iY + 1);
   602 		rightCol = Min(aGlyphImageSize.iWidth, aPos.iY - aClipRect.iTl.iY + 1);
   603 		TInt y = aPos.iY - (rightCol - 1);
   604 		for (TInt col = rightCol - 1; col >= leftCol; col--, y++)
   605 			{
   606 			TInt x = aPos.iX + topRow;
   607 			for (TInt row = topRow; row < bottomRow; row += KBufferSize, x += KBufferSize)
   608 				{
   609 				TInt length = KBufferSize;
   610 				if (length > bottomRow - row)
   611 					{
   612 					length = bottomRow - row;
   613 					}
   614 				const TUint8* p = aGlyphImage + row * aGlyphImageSize.iWidth + col;
   615 				for (TInt i = 0; i < length; i++, p += aGlyphImageSize.iWidth)
   616 					{
   617 					maskBuffer[i] = *p;
   618 					}
   619 				iDrawDevice->WriteRgbAlphaMulti(x, y, length, iPenColor, maskBuffer);
   620 				}
   621 			}
   622 		}
   623 	else
   624 		{
   625 		topRow = Max(0, aPos.iX - aClipRect.iBr.iX + 1);
   626 		bottomRow = Min(aGlyphImageSize.iHeight, aPos.iX - aClipRect.iTl.iX + 1);
   627 		leftCol = Max(0, aClipRect.iTl.iY - aPos.iY);
   628 		rightCol = Min(aGlyphImageSize.iWidth, aClipRect.iBr.iY - aPos.iY);
   629 		int y = aPos.iY + leftCol;
   630 		for (TInt col = leftCol; col < rightCol; col++, y++)
   631 			{
   632 			TInt x = aPos.iX - (bottomRow - 1);
   633 			for (TInt row = bottomRow; row > topRow; row -= KBufferSize, x += KBufferSize)
   634 				{
   635 				int length = KBufferSize;
   636 				if (length > row - topRow)
   637 					length = row - topRow;
   638 				const TUint8* p = aGlyphImage + (row - 1) * aGlyphImageSize.iWidth + col;
   639 				for (TInt i = 0; i < length; i++, p -= aGlyphImageSize.iWidth)
   640 					{
   641 					maskBuffer[i] = *p;
   642 					}
   643 				iDrawDevice->WriteRgbAlphaMulti(x, y, length, iPenColor, maskBuffer);
   644 				}
   645 			}
   646 		}
   647 	}
   648 
   649 
   650 /**
   651 Draw a four colour glyph.
   652 	
   653 @param	aPos			Position to start drawing gyph.
   654 @param	aGlyphImage		Pointer to the glyph image data.
   655 @param	aGlyphImageSize	Glyph image size.
   656 @param	aClipRect		Clipping rect.
   657 */
   658 void CSwDirectGdiEngine::DrawFourColourGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
   659 											 const TSize& aGlyphImageSize, const TRect& aClipRect)
   660 	{
   661 	const TInt topRow = Max(0, aClipRect.iTl.iY - aPos.iY);
   662 	const TInt bottomRow = Min(aGlyphImageSize.iHeight, aClipRect.iBr.iY - aPos.iY);
   663 	const TInt leftCol = Max(0, aClipRect.iTl.iX - aPos.iX);
   664 	const TInt rightCol = Min(aGlyphImageSize.iWidth, aClipRect.iBr.iX - aPos.iX);
   665 	const TUint8* p = aGlyphImage + (topRow * aGlyphImageSize.iWidth) + leftCol;
   666 	const TInt x = aPos.iX + leftCol;
   667 	TInt y = aPos.iY + topRow;
   668 	const TInt cols = rightCol - leftCol;
   669 	const TUint32 penColor = iPenColor.Internal();
   670 	const TUint32 shadowColor = iTextShadowColor.Internal();
   671 	const TUint32 brushColor = iBrushColor.Internal();
   672 
   673 	MOutlineAndShadowBlend* outlineAndShadow = NULL;
   674 	const TInt err = iDrawDevice->GetInterface(KOutlineAndShadowInterfaceID, reinterpret_cast <TAny*&> (outlineAndShadow));
   675 	if (err == KErrNone) 
   676 		{
   677 		//There is a support for the interface with KOutlineAndShadowInterface id.
   678 		for (TInt row = topRow; row < bottomRow; row++, p += aGlyphImageSize.iWidth, y++)
   679 			{
   680 			outlineAndShadow->WriteRgbOutlineAndShadow(x, y, cols, penColor, shadowColor, brushColor, p);
   681 			}
   682 		}
   683 	else
   684 		{
   685 		// Assert if MOutlineAndShadowBlend interface is not implemented
   686 		GRAPHICS_ASSERT_DEBUG(outlineAndShadow, EDirectGdiPanicInvalidInterfaceHandle);
   687 		}
   688 	}
   689 
   690 
   691 /**
   692 Draw a rotated four colour glyph.
   693 	
   694 @param	aPos			Position to start drawing gyph after rotation has been applied.
   695 @param	aGlyphImage		Pointer to the glyph image data.
   696 @param	aGlyphImageSize	Glyph image size.
   697 @param	aClipRect		Clipping rect.
   698 @param	aRotation		Rotation specifying how the glyph will be drawn.
   699 */
   700 void CSwDirectGdiEngine::DrawRotatedFourColourGlyph(const TPoint& aPos, const TUint8* aGlyphImage, const TSize& aGlyphImageSize,
   701 													const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
   702 	{
   703 	const int KBufferSize = 32;
   704 	TUint8 maskBuffer[KBufferSize];
   705 	int topRow = 0;
   706 	int bottomRow = 0;
   707 	int leftCol = 0;
   708 	int rightCol = 0;
   709 	const TUint32 penColor = iPenColor.Internal();
   710 	const TUint32 shadowColor = iTextShadowColor.Internal();
   711 	const TUint32 brushColor = iBrushColor.Internal();
   712 	
   713 	MOutlineAndShadowBlend* outlineAndShadow = NULL;
   714 	TInt err = iDrawDevice->GetInterface(KOutlineAndShadowInterfaceID, reinterpret_cast <TAny*&> (outlineAndShadow));
   715 	if(err != KErrNone)
   716 		{
   717 		// Assert if MOutlineAndShadowBlend interface is not implemented
   718 		GRAPHICS_ASSERT_DEBUG(outlineAndShadow, EDirectGdiPanicInvalidInterfaceHandle);
   719 		}
   720 	
   721 	if (aRotation == DirectGdi::EGraphicsRotation270)
   722 		{
   723 		topRow = Max(0, aClipRect.iTl.iX - aPos.iX);
   724 		bottomRow = Min(aGlyphImageSize.iHeight, aClipRect.iBr.iX - aPos.iX);
   725 		leftCol = Max(0, aPos.iY - aClipRect.iBr.iY + 1);
   726 		rightCol = Min(aGlyphImageSize.iWidth, aPos.iY - aClipRect.iTl.iY + 1);
   727 		TInt y = aPos.iY - (rightCol - 1);
   728 		for (TInt col = rightCol - 1; col >= leftCol; col--, y++)
   729 			{
   730 			TInt x = aPos.iX + topRow;
   731 			for (TInt row = topRow; row < bottomRow; row += KBufferSize, x += KBufferSize)
   732 				{
   733 				TInt length = KBufferSize;
   734 				if (length > bottomRow - row)
   735 					{
   736 					length = bottomRow - row;
   737 					}
   738 				const TUint8* p = aGlyphImage + row * aGlyphImageSize.iWidth + col;
   739 				for (TInt i = 0; i < length; i++, p += aGlyphImageSize.iWidth)
   740 					{
   741 					maskBuffer[i] = *p;
   742 					}
   743 				//There is a support for the interface with KOutlineAndShadowInterface id.
   744 				outlineAndShadow->WriteRgbOutlineAndShadow(x, y, length, penColor, shadowColor,	brushColor, maskBuffer);
   745 				}
   746 			}
   747 		}
   748 	else
   749 		{
   750 		topRow = Max(0, aPos.iX - aClipRect.iBr.iX + 1);
   751 		bottomRow = Min(aGlyphImageSize.iHeight, aPos.iX - aClipRect.iTl.iX + 1);
   752 		leftCol = Max(0, aClipRect.iTl.iY - aPos.iY);
   753 		rightCol = Min(aGlyphImageSize.iWidth, aClipRect.iBr.iY - aPos.iY);
   754 		int y = aPos.iY + leftCol;
   755 		for (TInt col = leftCol; col < rightCol; col++, y++)
   756 			{
   757 			TInt x = aPos.iX - (bottomRow - 1);
   758 			for (TInt row = bottomRow; row > topRow; row -= KBufferSize, x += KBufferSize)
   759 				{
   760 				int length = KBufferSize;
   761 				if (length > row - topRow)
   762 					length = row - topRow;
   763 				const TUint8* p = aGlyphImage + (row - 1) * aGlyphImageSize.iWidth + col;
   764 				for (TInt i = 0; i < length; i++, p -= aGlyphImageSize.iWidth)
   765 					{
   766 					maskBuffer[i] = *p;
   767 					}
   768 				//There is a support for the interface with KOutlineAndShadowInterface id.
   769 				outlineAndShadow->WriteRgbOutlineAndShadow(x, y, length, penColor, shadowColor,	brushColor, maskBuffer);
   770 				}
   771 			}
   772 		}
   773 	}
   774 
   775 
   776 /**
   777 Helper function to clip the array to the clip rect.
   778 
   779 @param	aArray		Start of array of data to be clipped.
   780 @param	aArrayLimit	End of array of data to be clipped.
   781 @param	aDataWd 	Length of data.
   782 @param	aDataHt		Height of data.
   783 @param	aPos		Position to start drawing from.
   784 @param	aClipRect	Rectangle to clip data array to.
   785 
   786 @return Pointer to array of clipped data.
   787 */
   788 TUint32* CSwDirectGdiEngine::ClipBinaryArray(TUint32* aArray, TUint32* aArrayLimit, TInt& aDataWd, TInt& aDataHt, TPoint& aPos, const TRect& aClipRect)
   789 	{
   790 	TUint32* arrayPtr = aArray;
   791 	TInt clipDiff = aClipRect.iTl.iX - aPos.iX;
   792 	if (clipDiff > 0)
   793 		{
   794 		while (arrayPtr < aArrayLimit)
   795 			{
   796 			*arrayPtr++ >>= clipDiff;
   797 			}
   798 		aDataWd -= clipDiff;
   799 		aPos.iX = aClipRect.iTl.iX;
   800 		arrayPtr = aArray;
   801 		}
   802 	if ((aPos.iX + aDataWd > aClipRect.iBr.iX) && (aDataWd > 0))
   803 		{
   804 		aDataWd = aClipRect.iBr.iX - aPos.iX;
   805 		}
   806 	clipDiff = aClipRect.iTl.iY - aPos.iY;
   807 	if (clipDiff > 0)
   808 		{
   809 		aDataHt -= clipDiff;
   810 		arrayPtr += clipDiff;
   811 		aPos.iY = aClipRect.iTl.iY;
   812 		}
   813 	if (((aPos.iY + aDataHt) > (aClipRect.iBr.iY)) && (aDataHt > 0))
   814 		{
   815 		aDataHt = aClipRect.iBr.iY - aPos.iY;
   816 		}
   817 	return arrayPtr;
   818 	}