os/graphics/graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgditext.cpp
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
16 #include "swdirectgdiengine.h"
17 #include <bitdrawinterfaceid.h>
18 #include <bmalphablend.h>
21 @see MDirectGdiEngine::SetFont()
23 void CSwDirectGdiEngine::SetFont(TUint32 /*aFontId*/)
26 // The SW adapter doesn't need the font ID to index a separate font cache because it uses the standard one.
31 @see MDirectGdiEngine::ResetFont()
33 void CSwDirectGdiEngine::ResetFont()
39 @see MDirectGdiEngine::SetTextShadowColor()
41 void CSwDirectGdiEngine::SetTextShadowColor(const TRgb& aColor)
43 iTextShadowColor = aColor;
47 @see MDirectGdiEngine::BeginDrawGlyph()
49 void CSwDirectGdiEngine::BeginDrawGlyph()
54 @see MDirectGdiEngine::DrawGlyph()
55 @panic DGDIAdapter 56, if an invalid glyph bitmap type is passed in.
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)
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++)
70 // Do the clip rects intersect?
71 regionRect = (*iDefaultRegionPtr)[count];
72 if (!regionRect.Intersects(clipRect))
77 // Clip to intersection of two clip rects
78 regionRect.Intersection(clipRect);
80 if (aRotation == DirectGdi::EGraphicsRotationNone) // Horizontal text
82 // Do the glyph and the clip rect intersect?
83 TRect glyphRect(pos, aGlyphImageSize);
84 if (!regionRect.Intersects(glyphRect))
89 // Clip to intersection with glyph bitmap
90 regionRect.Intersection(glyphRect);
92 switch (aGlyphBitmapType)
94 case EMonochromeGlyphBitmap:
96 DrawBitmapGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
99 case EAntiAliasedGlyphBitmap:
101 DrawAntiAliasedGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
104 case EFourColourBlendGlyphBitmap:
106 DrawFourColourGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect);
110 GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidGlyphBitmapType);
113 else // Vertical text
116 // Do the glyph and the clip rect intersect?
117 TRect glyphRect(aPos, aGlyphImageSize);
118 if (!regionRect.Intersects(glyphRect))
123 // Clip to intersection with glyph bitmap
124 regionRect.Intersection(glyphRect);
127 switch (aGlyphBitmapType)
129 case EMonochromeGlyphBitmap:
131 DrawRotatedBitmapGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
134 case EAntiAliasedGlyphBitmap:
136 DrawRotatedAntiAliasedGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
139 case EFourColourBlendGlyphBitmap:
141 DrawRotatedFourColourGlyph(pos, aGlyphImage, aGlyphImageSize, regionRect, aRotation);
145 GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidGlyphBitmapType);
149 iDrawDevice->UpdateRegion(regionRect);
154 @see MDirectGdiEngine::EndDrawGlyph()
156 void CSwDirectGdiEngine::EndDrawGlyph()
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.
168 void CSwDirectGdiEngine::DrawBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
169 const TSize& aGlyphImageSize, const TRect& aClipRect)
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
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).
179 TInt dataHeight = aGlyphImageSize.iHeight;
180 TInt dataLength = aGlyphImageSize.iWidth;
181 if (dataLength > 30 || dataHeight > 32)
183 DrawLargeBitmapGlyph(aPos, aGlyphImage, aGlyphImageSize, aClipRect);
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
194 repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
195 repeatCount >>= bitIndex & 7;
196 TInt multiLineFlag = repeatCount & 1;
200 binaryDataPtrLimit = binaryData + glyphLine + repeatCount;
203 while (binaryDataPtr < binaryDataPtrLimit)
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;
212 *binaryDataPtr |= (*glyphDataWord << (32 - bitShift));
214 bitIndex += dataLength;
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;
227 data |= (*glyphDataWord << (32 - bitShift));
229 while (binaryDataPtr < binaryDataPtrLimit)
231 *binaryDataPtr++ = data;
233 bitIndex += dataLength;
236 TPoint topLeft(aPos);
237 binaryDataPtr = ClipBinaryArray(binaryData, binaryData + dataHeight, dataLength, dataHeight, topLeft, aClipRect);
238 if ((dataLength > 0) && (dataHeight > 0))
240 iDrawDevice->WriteBinary(topLeft.iX, topLeft.iY, binaryDataPtr, dataLength, dataHeight, iPenColor, GcDrawMode(iDrawMode) );
246 Draw a large bitmap glyph.
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.
253 void CSwDirectGdiEngine::DrawLargeBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
254 const TSize& aGlyphImageSize, const TRect& aClipRect)
256 TPoint printPos(aPos);
257 const TInt dataLength = aGlyphImageSize.iWidth;
258 const TInt dataHeight = aGlyphImageSize.iHeight;
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
265 repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
266 repeatCount >>= bitIndex & 7;
267 const TInt multiLineFlag = repeatCount & 1;
273 for (TInt currentLine = 0; currentLine < repeatCount; currentLine++)
275 CopyCharLine(scanLineBuffer, scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
276 OutputCharLineMultiplied(printPos, scanLineBuffer, dataLength, 1, aClipRect);
277 bitIndex += dataLength;
283 CopyCharLine(scanLineBuffer, scanLineWords, aGlyphImage + (bitIndex >> 3), bitIndex & 7, dataLength);
284 OutputCharLineMultiplied(printPos, scanLineBuffer, dataLength, repeatCount, aClipRect);
285 printPos.iY += repeatCount;
286 bitIndex += dataLength;
295 void CSwDirectGdiEngine::CopyCharLine(TUint32* aBinaryDataPtr, TInt aBufferWords, const TUint8* aData, TInt aBitShift, TInt aCharWidth)
298 TInt wordsToCopy = (aCharWidth + 31) >> 5;
299 if (wordsToCopy > aBufferWords)
301 wordsToCopy = aBufferWords;
303 TUint32* ptrLimit = aBinaryDataPtr + wordsToCopy;
304 TUint32* dataWord = (TUint32*)(TInt(aData) & ~3);
305 aBitShift += (TInt(aData) - TInt(dataWord)) << 3;
306 while (aBinaryDataPtr < ptrLimit)
308 *aBinaryDataPtr = *dataWord++;
309 *aBinaryDataPtr >>= aBitShift;
312 *aBinaryDataPtr |= (*dataWord << (32 - aBitShift));
322 void CSwDirectGdiEngine::OutputCharLineMultiplied(TPoint aPrintPos, TUint32* aBuffer, TInt aDataLength, TInt aNum, const TRect& aClipRect)
324 if (aDataLength <= 0)
328 TInt bufferWords = (aDataLength + 31) >> 5;
329 TUint32* bufferLimit = aBuffer + bufferWords;
330 if (aPrintPos.iX < aClipRect.iTl.iX)
332 TInt pixelExcess = aClipRect.iTl.iX - aPrintPos.iX;
333 while (pixelExcess >= 32)
340 if (aDataLength <= 0)
346 TInt shiftUp = 32 - pixelExcess;
347 TUint32* bufferPtr = aBuffer;
348 while (bufferPtr < bufferLimit)
350 *bufferPtr >>= pixelExcess;
351 if (bufferPtr < bufferLimit - 1)
353 *bufferPtr |= (*(bufferPtr + 1) << shiftUp);
357 aDataLength -= pixelExcess;
358 if (aDataLength <= 0)
363 aPrintPos.iX = aClipRect.iTl.iX;
365 if (aPrintPos.iX + aDataLength > aClipRect.iBr.iX)
367 TInt pixelExcess = aPrintPos.iX + aDataLength - aClipRect.iBr.iX;
368 aDataLength -= pixelExcess;
369 if (aDataLength <= 0)
376 if ((aPrintPos.iY >= aClipRect.iTl.iY) && (aPrintPos.iY < aClipRect.iBr.iY))
378 iDrawDevice->WriteBinaryLine(aPrintPos.iX, aPrintPos.iY, aBuffer, aDataLength, iPenColor, GcDrawMode(iDrawMode));
387 Draw a rotated bitmap glyph.
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.
395 void CSwDirectGdiEngine::DrawRotatedBitmapGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
396 const TSize& aGlyphImageSize, const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
398 TPoint printPos(aPos);
399 TInt dataLength = aGlyphImageSize.iWidth;
400 TInt dataHeight = aGlyphImageSize.iHeight;
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...
408 repeatCount = Load16(aGlyphImage + (bitIndex >> 3));
409 repeatCount >>= bitIndex & 7;
410 TInt multiLineFlag = repeatCount & 1;
414 TInt signedRepeatCount = repeatCount * direction;
417 for (TInt currentLine = 0; currentLine < repeatCount; currentLine++)
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;
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;
439 void CSwDirectGdiEngine::OutputCharLineVertical(TPoint aPrintPos, TUint32* aBuffer, TInt aDataLength,
440 TInt aNum, TInt aDirection, const TRect& aClipRect)
442 // const DirectGdi::TGraphicsRotation rotation = aUp ? DirectGdi::EGraphicsRotation270 : DirectGdi::EGraphicsRotation90;
443 // TInt direction = (aRotation == DirectGdi::EGraphicsRotation270) ? 1 : -1;
444 if (aDataLength <= 0)
448 TInt bufferWords = (aDataLength + 31) >> 5;
449 TUint32* bufferLimit = aBuffer + bufferWords;
452 if (aPrintPos.iY >= aClipRect.iBr.iY)
454 TInt pixelExcess = aPrintPos.iY - aClipRect.iBr.iY + 1;
455 while (pixelExcess >= 32)
461 if (aDataLength <= 0)
467 TInt shiftUp = 32 - pixelExcess;
468 TUint32* bufferPtr = aBuffer;
469 while (bufferPtr < bufferLimit)
471 *bufferPtr >>= pixelExcess;
472 if (bufferPtr < bufferLimit - 1)
473 *bufferPtr |= (*(bufferPtr + 1) << shiftUp);
476 aDataLength -= pixelExcess;
477 if (aDataLength <= 0)
482 aPrintPos.iY = aClipRect.iBr.iY - 1;
484 if ((aPrintPos.iY - aDataLength) < (aClipRect.iTl.iY - 1))
486 TInt pixelExcess = aClipRect.iTl.iY - 1 - aPrintPos.iY + aDataLength;
487 aDataLength -= pixelExcess;
488 if (aDataLength <= 0)
496 if (aPrintPos.iY < aClipRect.iTl.iY)
498 TInt pixelExcess = aClipRect.iTl.iY - aPrintPos.iY;
499 while (pixelExcess >= 32)
505 if (aDataLength <= 0)
511 TInt shiftup = 32 - pixelExcess;
512 TUint32* bufferptr = aBuffer;
513 while (bufferptr < bufferLimit)
515 *bufferptr >>= pixelExcess;
516 if (bufferptr < bufferLimit - 1)
517 *bufferptr |= (*(bufferptr + 1) << shiftup);
520 aDataLength -= pixelExcess;
521 if (aDataLength <= 0)
526 aPrintPos.iY = aClipRect.iTl.iY;
528 if (aPrintPos.iY + aDataLength > aClipRect.iBr.iY)
530 TInt pixelExcess = aPrintPos.iY + aDataLength - aClipRect.iBr.iY;
531 aDataLength -= pixelExcess;
532 if (aDataLength <= 0)
538 CGraphicsContext::TDrawMode drawMode = GcDrawMode(iDrawMode);
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;
550 Draw an antialiased glyph.
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.
557 void CSwDirectGdiEngine::DrawAntiAliasedGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
558 const TSize& aGlyphImageSize, const TRect& aClipRect)
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;
569 for (TInt row = topRow; row < bottomRow; row++, p += aGlyphImageSize.iWidth, y++)
571 iDrawDevice->WriteRgbAlphaMulti(x, y, cols, iPenColor, p);
577 Draw a rotated antialiased glyph.
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.
585 void CSwDirectGdiEngine::DrawRotatedAntiAliasedGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
586 const TSize& aGlyphImageSize, const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
588 const int KBufferSize = 32;
589 TUint8 maskBuffer[KBufferSize];
594 const TUint32 penColor = iPenColor.Internal();
595 const TUint32 brushColor = iBrushColor.Internal();
597 if (aRotation == DirectGdi::EGraphicsRotation270)
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++)
606 TInt x = aPos.iX + topRow;
607 for (TInt row = topRow; row < bottomRow; row += KBufferSize, x += KBufferSize)
609 TInt length = KBufferSize;
610 if (length > bottomRow - row)
612 length = bottomRow - row;
614 const TUint8* p = aGlyphImage + row * aGlyphImageSize.iWidth + col;
615 for (TInt i = 0; i < length; i++, p += aGlyphImageSize.iWidth)
619 iDrawDevice->WriteRgbAlphaMulti(x, y, length, iPenColor, maskBuffer);
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++)
632 TInt x = aPos.iX - (bottomRow - 1);
633 for (TInt row = bottomRow; row > topRow; row -= KBufferSize, x += KBufferSize)
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)
643 iDrawDevice->WriteRgbAlphaMulti(x, y, length, iPenColor, maskBuffer);
651 Draw a four colour glyph.
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.
658 void CSwDirectGdiEngine::DrawFourColourGlyph(const TPoint& aPos, const TUint8* aGlyphImage,
659 const TSize& aGlyphImageSize, const TRect& aClipRect)
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();
673 MOutlineAndShadowBlend* outlineAndShadow = NULL;
674 const TInt err = iDrawDevice->GetInterface(KOutlineAndShadowInterfaceID, reinterpret_cast <TAny*&> (outlineAndShadow));
677 //There is a support for the interface with KOutlineAndShadowInterface id.
678 for (TInt row = topRow; row < bottomRow; row++, p += aGlyphImageSize.iWidth, y++)
680 outlineAndShadow->WriteRgbOutlineAndShadow(x, y, cols, penColor, shadowColor, brushColor, p);
685 // Assert if MOutlineAndShadowBlend interface is not implemented
686 GRAPHICS_ASSERT_DEBUG(outlineAndShadow, EDirectGdiPanicInvalidInterfaceHandle);
692 Draw a rotated four colour glyph.
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.
700 void CSwDirectGdiEngine::DrawRotatedFourColourGlyph(const TPoint& aPos, const TUint8* aGlyphImage, const TSize& aGlyphImageSize,
701 const TRect& aClipRect, const DirectGdi::TGraphicsRotation aRotation)
703 const int KBufferSize = 32;
704 TUint8 maskBuffer[KBufferSize];
709 const TUint32 penColor = iPenColor.Internal();
710 const TUint32 shadowColor = iTextShadowColor.Internal();
711 const TUint32 brushColor = iBrushColor.Internal();
713 MOutlineAndShadowBlend* outlineAndShadow = NULL;
714 TInt err = iDrawDevice->GetInterface(KOutlineAndShadowInterfaceID, reinterpret_cast <TAny*&> (outlineAndShadow));
717 // Assert if MOutlineAndShadowBlend interface is not implemented
718 GRAPHICS_ASSERT_DEBUG(outlineAndShadow, EDirectGdiPanicInvalidInterfaceHandle);
721 if (aRotation == DirectGdi::EGraphicsRotation270)
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++)
730 TInt x = aPos.iX + topRow;
731 for (TInt row = topRow; row < bottomRow; row += KBufferSize, x += KBufferSize)
733 TInt length = KBufferSize;
734 if (length > bottomRow - row)
736 length = bottomRow - row;
738 const TUint8* p = aGlyphImage + row * aGlyphImageSize.iWidth + col;
739 for (TInt i = 0; i < length; i++, p += aGlyphImageSize.iWidth)
743 //There is a support for the interface with KOutlineAndShadowInterface id.
744 outlineAndShadow->WriteRgbOutlineAndShadow(x, y, length, penColor, shadowColor, brushColor, maskBuffer);
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++)
757 TInt x = aPos.iX - (bottomRow - 1);
758 for (TInt row = bottomRow; row > topRow; row -= KBufferSize, x += KBufferSize)
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)
768 //There is a support for the interface with KOutlineAndShadowInterface id.
769 outlineAndShadow->WriteRgbOutlineAndShadow(x, y, length, penColor, shadowColor, brushColor, maskBuffer);
777 Helper function to clip the array to the clip rect.
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.
786 @return Pointer to array of clipped data.
788 TUint32* CSwDirectGdiEngine::ClipBinaryArray(TUint32* aArray, TUint32* aArrayLimit, TInt& aDataWd, TInt& aDataHt, TPoint& aPos, const TRect& aClipRect)
790 TUint32* arrayPtr = aArray;
791 TInt clipDiff = aClipRect.iTl.iX - aPos.iX;
794 while (arrayPtr < aArrayLimit)
796 *arrayPtr++ >>= clipDiff;
799 aPos.iX = aClipRect.iTl.iX;
802 if ((aPos.iX + aDataWd > aClipRect.iBr.iX) && (aDataWd > 0))
804 aDataWd = aClipRect.iBr.iX - aPos.iX;
806 clipDiff = aClipRect.iTl.iY - aPos.iY;
810 arrayPtr += clipDiff;
811 aPos.iY = aClipRect.iTl.iY;
813 if (((aPos.iY + aDataHt) > (aClipRect.iBr.iY)) && (aDataHt > 0))
815 aDataHt = aClipRect.iBr.iY - aPos.iY;