javauis/lcdui_akn/lcdgr/src/CMIDGraphics.cpp
branchRCL_3
changeset 26 2455ef1f5bbc
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/javauis/lcdui_akn/lcdgr/src/CMIDGraphics.cpp	Wed Sep 01 12:33:18 2010 +0100
@@ -0,0 +1,742 @@
+/*
+* Copyright (c) 2005 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:
+*
+*/
+
+#include <fbs.h>
+#include <bitdev.h>
+#include <gdi.h>
+
+#ifdef RD_JAVA_NGA_ENABLED
+#include <coecntrl.h>
+#include <e32cmn.h>
+#include <AknLayoutFont.h>
+#include <lcdgdrv.h> // for TTransformType
+#endif // RD_JAVA_NGA_ENABLED
+
+#include "fbslcdgraphics.h"
+
+#include "CMIDGraphics.h"
+#include "CMIDImage.h"
+
+#include "MidProxyMap.h"
+#include "LcdGraphics.h"
+#include "LcdImage.h"
+#include "LcdSurface.h"
+#include "LcdBitmapSurface.h"
+#include "LcdWindowSurface.h"
+
+#define UnhandAddress(aAddress) (aAddress << 2)
+
+template <class T>
+inline T* Unhand(TInt aAddress)
+{
+    return reinterpret_cast<T*>(UnhandAddress(aAddress));
+}
+
+// ---------------------------------------------------------------------------
+// Constructor
+// ---------------------------------------------------------------------------
+//
+#ifdef RD_JAVA_NGA_ENABLED
+CMIDGraphics::CMIDGraphics(
+    CMIDProxyMap& aProxyMap, CLcdGraphics* aGraphics,
+    TBool aIsImageTarget, MMIDCanvas* aTarget)
+        : iCanvasTarget(aTarget)
+        , iProxyMap(aProxyMap)
+        , iGraphics(aGraphics)
+        , iIsImageTarget(aIsImageTarget)
+{
+}
+#else // !RD_JAVA_NGA_ENABLED
+CMIDGraphics::CMIDGraphics(
+    CMIDProxyMap& aProxyMap, CLcdGraphics* aGraphics, TBool aIsImageTarget)
+        : iProxyMap(aProxyMap)
+        , iGraphics(aGraphics)
+        , iIsImageTarget(aIsImageTarget)
+{
+}
+#endif // RD_JAVA_NGA_ENABLED
+
+// ---------------------------------------------------------------------------
+// Destructor
+// ---------------------------------------------------------------------------
+//
+CMIDGraphics::~CMIDGraphics()
+{
+    delete iGraphics;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::Type()
+// ---------------------------------------------------------------------------
+//
+MMIDComponent::TType CMIDGraphics::Type() const
+{
+    return EGraphics;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::Dispose()
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::Dispose()
+{
+    delete this;
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDBufferProcessor
+// CMIDGraphics::Processor()
+// ---------------------------------------------------------------------------
+//
+MMIDBufferProcessor* CMIDGraphics::Processor()
+{
+    return this;
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDBufferProcessor
+// CMIDGraphics::AbortAsync()
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::AbortAsync()
+{
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDBufferProcessor
+// CMIDGraphics::ProcessL()
+// ---------------------------------------------------------------------------
+//
+#ifdef RD_JAVA_NGA_ENABLED
+TBool CMIDGraphics::ProcessL(
+    const TMIDBufferOp*& aRead, const TMIDBufferOp* aEnd,
+    TInt& /*aCycles*/, java::util::Monitor* /*aMonitor*/)
+{
+    // Checks if EGL is availabe.
+    TBool eglAvailable = EFalse;
+    if (iCanvasTarget && iCanvasTarget->IsEglAvailable())
+    {
+        // Egl is available so bounding boxes of 2D primitives
+        // need to be calculated.
+        eglAvailable = ETrue;
+    }
+#else // !RD_JAVA_NGA_ENABLED
+TBool CMIDGraphics::ProcessL(
+    const TMIDBufferOp*& aRead, const TMIDBufferOp* aEnd,
+    TInt& /*aCycles*/, TRequestStatus* /*aStatus*/)
+{
+#endif // RD_JAVA_NGA_ENABLED   
+    CLcdGraphics& graphics = *iGraphics;
+
+    graphics.Begin();
+    while (aRead < aEnd)
+    {
+        TInt opCode = aRead->OpCode();
+
+        switch (opCode)
+        {
+        case EGrOpSetColor:
+        {
+            const TLcdGrOpSetColor* setColor = (const TLcdGrOpSetColor*)(aRead);
+            graphics.SetColor(setColor->iColor);
+        }
+        break;
+
+        case EGrOpSetStroke:
+        {
+            const TLcdGrOpSetStrokeStyle* setStroke =
+                (const TLcdGrOpSetStrokeStyle*)(aRead);
+            graphics.SetStrokeStyle((TStrokeStyle)setStroke->iStyle);
+        }
+        break;
+
+        case EGrOpSetFont:
+        {
+            const TLcdGrOpSetFont* setFont = (const TLcdGrOpSetFont*)(aRead);
+            graphics.SetFont(setFont->iFont);
+        }
+        break;
+
+        case EGrOpSetClipRect:
+        {
+            const TLcdGrOpSetClipRect* clipRect= (const TLcdGrOpSetClipRect*)(aRead);
+            graphics.SetClipRect(clipRect->iPosition, clipRect->iSize);
+        }
+        break;
+
+        case EGrOpTranslate:
+        {
+            const TLcdGrOpTranslate* translate = (const TLcdGrOpTranslate*)(aRead);
+            graphics.Translate(translate->iVector);
+        }
+        break;
+
+        case EGrOpReset:
+        {
+            const TLcdGrOpReset* reset = (const TLcdGrOpReset*)(aRead);
+            graphics.Reset(reset->iSize);
+        }
+        break;
+
+        case EGrOpDrawLine:
+        {
+            ASSERT(aRead->Size() >= 5);
+            const TLcdGrOpDrawLine* line = (const TLcdGrOpDrawLine*)(aRead);
+            graphics.DrawLine(line->iPoints[0], line->iPoints[1]);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                // Update bounding box
+                TRect rect(line->iPoints[0], line->iPoints[1]);
+                rect.Normalize();
+                // Move br by one pixel to include the end point
+                ++rect.iBr.iX;
+                ++rect.iBr.iY;
+                UpdateRect(rect);
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawRect:
+        {
+            const TLcdGrOpDrawRect* drawRect = (const TLcdGrOpDrawRect*)(aRead);
+            graphics.DrawRect(drawRect->iPosition, drawRect->iSize);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               drawRect->iPosition,
+                               TSize(drawRect->iSize.iWidth + 1, drawRect->iSize.iHeight + 1)));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpFillRect:
+        {
+            const TLcdGrOpFillRect* fillRect = (const TLcdGrOpFillRect*)(aRead);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                TRect rect(fillRect->iPosition, fillRect->iSize);
+                if (!FillEglSurface(rect, TRgb::Color16MA(graphics.Color())))
+                {
+                    graphics.FillRect(fillRect->iPosition, fillRect->iSize);
+                    UpdateRect(TRect(
+                                   fillRect->iPosition,
+                                   TSize(fillRect->iSize.iWidth + 1,
+                                         fillRect->iSize.iHeight + 1)));
+                }
+            }
+            else
+            {
+                graphics.FillRect(fillRect->iPosition, fillRect->iSize);
+            }
+#else // !RD_JAVA_NGA_ENABLED
+            graphics.FillRect(fillRect->iPosition, fillRect->iSize);
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawRoundRect:
+        {
+            const TLcdGrOpDrawRoundRect* drawRoundRect =
+                (const TLcdGrOpDrawRoundRect*)(aRead);
+            graphics.DrawRoundRect(
+                drawRoundRect->iPosition, drawRoundRect->iSize,
+                drawRoundRect->iCornerSize);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               drawRoundRect->iPosition,
+                               TSize(drawRoundRect->iSize.iWidth + 1,
+                                     drawRoundRect->iSize.iHeight + 1)));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpFillRoundRect:
+        {
+            const TLcdGrOpFillRoundRect* fillRoundRect =
+                (const TLcdGrOpFillRoundRect*)(aRead);
+            graphics.FillRoundRect(
+                fillRoundRect->iPosition, fillRoundRect->iSize,
+                fillRoundRect->iCornerSize);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               fillRoundRect->iPosition,
+                               TSize(fillRoundRect->iSize.iWidth + 1,
+                                     fillRoundRect->iSize.iHeight + 1)));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawArc:
+        {
+            const TLcdGrOpDrawArc* drawArc = (const TLcdGrOpDrawArc*)(aRead);
+            graphics.DrawArc(
+                drawArc->iPosition, drawArc->iSize, drawArc->iStart,
+                drawArc->iAngle);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               drawArc->iPosition,
+                               TSize(drawArc->iSize.iWidth + 1, drawArc->iSize.iHeight + 1)));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpFillArc:
+        {
+            const TLcdGrOpFillArc* fillArc = (const TLcdGrOpFillArc*)(aRead);
+            graphics.FillArc(
+                fillArc->iPosition, fillArc->iSize, fillArc->iStart,
+                fillArc->iAngle);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               fillArc->iPosition,
+                               TSize(fillArc->iSize.iWidth + 1, fillArc->iSize.iHeight + 1)));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawText:
+        {
+            const TLcdGrOpDrawText* drawText = (const TLcdGrOpDrawText*)(aRead);
+            TPtrC ptr(&drawText->iText[0], drawText->iLength);
+            graphics.DrawText(ptr, drawText->iPoint, drawText->iAnchor);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateTextRect(ptr, drawText->iPoint, drawText->iAnchor);
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawImage:
+        {
+            const TLcdGrOpDrawImage* drawImage = (const TLcdGrOpDrawImage*)(aRead);
+            MMIDComponent* component = Unhand<MMIDComponent>(drawImage->iImage);
+            ASSERT(component);
+            const CMIDImage& proxy =
+                static_cast<const CMIDImage&>(*static_cast<MMIDImage*>(component));
+            const CLcdImage& image = proxy.Image();
+#ifndef RD_JAVA_NGA_ENABLED
+            graphics.DrawImage(image, drawImage->iPoint, drawImage->iAnchor);
+#else // RD_JAVA_NGA_ENABLED
+            graphics.DrawImage(image, drawImage->iPoint, drawImage->iAnchor, eglAvailable);
+            if (eglAvailable)
+            {
+                UpdateRect(CalcDstRect(
+                               drawImage->iPoint, image.Size(), ETransNone,
+                               (::TAnchor)drawImage->iAnchor));
+            }
+#endif // !RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpDrawRegion:
+        {
+            const TLcdGrOpDrawRegion* drawRegion = (const TLcdGrOpDrawRegion*)(aRead);
+            MMIDComponent* component = Unhand<MMIDComponent>(drawRegion->iImage);
+            ASSERT(component);
+            const CMIDImage& proxy =
+                static_cast<const CMIDImage&>(*static_cast<MMIDImage*>(component));
+            const CLcdImage& image = proxy.Image();
+#ifndef RD_JAVA_NGA_ENABLED
+            graphics.DrawRegion(
+                image, drawRegion->iSrcPosition, drawRegion->iSrcSize,
+                drawRegion->iTransform, drawRegion->iDstPoint,
+                drawRegion->iAnchor);
+#else // RD_JAVA_NGA_ENABLED
+            graphics.DrawRegion(
+                image, drawRegion->iSrcPosition, drawRegion->iSrcSize,
+                drawRegion->iTransform, drawRegion->iDstPoint,
+                drawRegion->iAnchor, eglAvailable);
+            if (eglAvailable)
+            {
+                UpdateRect(CalcDstRect(drawRegion->iDstPoint, drawRegion->iSrcSize,
+                                       (TTransformType)drawRegion->iTransform,
+                                       (::TAnchor)drawRegion->iAnchor));
+            }
+#endif // !RD_JAVA_NGA_ENABLED   
+        }
+        break;
+
+        case EGrOpCopyArea:
+        {
+            // NGA: EGrOpCopyArea is not valid for canvas
+            const TLcdGrOpCopyArea* copyArea = (const TLcdGrOpCopyArea*)(aRead);
+            graphics.CopyArea(copyArea->iSrcPosition, copyArea->iSrcSize,
+                              copyArea->iPoint, copyArea->iAnchor);
+        }
+        break;
+
+        case EGrOpFillTriangle:
+        {
+            ASSERT(aRead->Size() >= 5);
+            const TLcdGrOpFillTriangle* triangle = (const TLcdGrOpFillTriangle*)(aRead);
+            graphics.FillTriangle(triangle->iPoints);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(
+                               Min3(triangle->iPoints[ 0 ].iX, triangle->iPoints[ 1 ].iX,
+                                    triangle->iPoints[ 2 ].iX),
+                               Min3(triangle->iPoints[ 0 ].iY, triangle->iPoints[ 1 ].iY,
+                                    triangle->iPoints[ 2 ].iY),
+                               Max3(triangle->iPoints[ 0 ].iX, triangle->iPoints[ 1 ].iX,
+                                    triangle->iPoints[ 2 ].iX) + 1,
+                               Max3(triangle->iPoints[ 0 ].iY, triangle->iPoints[ 1 ].iY,
+                                    triangle->iPoints[ 2 ].iY) + 1));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+        case EGrOpFlush:
+        {
+            const TLcdGrOpFlush* flush = (const TLcdGrOpFlush*)(aRead);
+            graphics.Flush(flush->iPosition, flush->iSize);
+        }
+        break;
+        case EGrOpDrawBackground:
+        {
+            const TLcdGrOpDrawBackground* drawBackground =
+                (const TLcdGrOpDrawBackground*)(aRead);
+            MMIDCanvas* canvas = Unhand<MMIDCanvas>(drawBackground->iCanvas);
+            ASSERT(canvas);
+            graphics.DrawBackground(canvas, drawBackground->iPosition, drawBackground->iSize);
+#ifdef RD_JAVA_NGA_ENABLED
+            if (eglAvailable)
+            {
+                UpdateRect(TRect(drawBackground->iPosition, drawBackground->iSize));
+            }
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+        case EGrOpCopyGraphics:
+        {
+#ifdef RD_JAVA_NGA_ENABLED
+            const TLcdGrOpCopyGraphics* copyGraphics =
+                (const TLcdGrOpCopyGraphics*)(aRead);
+            MMIDComponent* component = Unhand<MMIDComponent>(copyGraphics->iSrcGraphics);
+            ASSERT(component);
+            CMIDGraphics* proxy =
+                static_cast<CMIDGraphics*>(static_cast<MMIDGraphics*>(component));
+            CLcdGraphics* srcGraphics = proxy->iGraphics;
+            proxy->UpdateOffScreenBitmapL(ETrue);
+            graphics.CopyGraphics(srcGraphics);
+
+#endif // RD_JAVA_NGA_ENABLED   
+        }
+        break;
+        default:
+#ifdef _TRACE
+            RDebug::Print(_L("CMIDGraphics::ProcessL - ignoring unknown opcode %d"), opCode);
+#endif
+            break;
+        }
+        aRead += aRead->Size();
+    }
+    graphics.End();
+    return EFalse;
+}
+
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::DrawPixels()
+// ---------------------------------------------------------------------------
+//
+TInt CMIDGraphics::DrawPixels
+(
+    TInt    aType,          // pixel data type
+    TUint8* aAddress,       // pixel data array
+    TInt    aLength,        // pixel array length - bytes
+    TInt    aScanLength,    // always positive - vertical reflection handled by point/size
+    TBool   aAlpha,         // true if pixel data contains alpha
+    const TRect& aRect,
+    TInt    aTransform      // transform to apply
+)
+{
+    TInt err;
+    TSize size(aRect.Size());
+    iGraphics->Begin();
+
+    TBool premultiplied(EFalse);
+#ifdef RD_JAVA_NGA_ENABLED
+    if (IsEglAvailable())
+    {
+        premultiplied = ETrue;
+        UpdateRect(aRect);
+    }
+#endif
+
+    err = iGraphics->DrawPixels(aType, aAddress, aLength, aScanLength,
+                                aAlpha, aRect.iTl, size, aTransform, premultiplied);
+    iGraphics->End();
+    return err;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::DisplayColor()
+// Maps a color provided as 32bit rgb triple to the closest color representable
+// in the target devices display mode. returns the result as a 32bit rgb triple.
+// ---------------------------------------------------------------------------
+//
+TUint32 CMIDGraphics::DisplayColor(TUint32 aRGB32)
+{
+    return iGraphics->QuantizeColor(aRGB32);
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::Bitmap()
+// Returns a pointer to the back buffer bitmap in use by the graphics object.
+// May return NULL.
+// ---------------------------------------------------------------------------
+//
+CFbsBitmap* CMIDGraphics::Bitmap() const
+{
+    CLcdSurface* surface = iGraphics->Surface();
+    if (surface->Type() == CLcdSurface::EBitmapSurface)
+    {
+        CLcdBitmapSurface* bitmapSurface = (CLcdBitmapSurface*)surface;
+        return bitmapSurface->Bitmap();
+    }
+    return NULL;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::Window()
+// ---------------------------------------------------------------------------
+//
+RWindow* CMIDGraphics::Window() const
+{
+    CLcdSurface* surface = iGraphics->Surface();
+    if (surface->Type() == CLcdSurface::EWindowSurface)
+    {
+        CLcdWindowSurface* windowSurface = (CLcdWindowSurface*)surface;
+        return windowSurface->Window();
+    }
+    return NULL;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::ViewPort()
+// @return drawing surface viewport rectangle.
+// ---------------------------------------------------------------------------
+//
+TRect CMIDGraphics::ViewPort() const
+{
+    return iGraphics->Surface()->Bounds();
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::IsImageTarget()
+// @return ETrue for mutable off-screen images (not for canvas/GameCanvas framebuffer)
+// or EFalse for canvas/GameCanvas framebuffer
+// ---------------------------------------------------------------------------
+//
+TBool CMIDGraphics::IsImageTarget() const
+{
+    return iIsImageTarget;
+}
+
+#ifdef RD_JAVA_NGA_ENABLED
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::UpdateEglContent()
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::UpdateEglContent()
+{
+    if (iCanvasTarget)
+    {
+        iCanvasTarget->UpdateEglContent();
+    }
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::IsEglAvailable()
+// ---------------------------------------------------------------------------
+//
+TBool CMIDGraphics::IsEglAvailable() const
+{
+    return (iCanvasTarget ? iCanvasTarget->IsEglAvailable() : EFalse);
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::BindEglSurface()
+// ---------------------------------------------------------------------------
+//
+EGLSurface CMIDGraphics::BindEglSurface()
+{
+    return (iCanvasTarget ? iCanvasTarget->BindEglSurface() : EGL_NO_SURFACE);
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::ReleaseEglSurface()
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::ReleaseEglSurface()
+{
+    if (iCanvasTarget)
+    {
+        iCanvasTarget->ReleaseEglSurface();
+    }
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::UpdateOffScreenBitmapL()
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::UpdateOffScreenBitmapL(TBool aForced)
+{
+    if (iCanvasTarget)
+    {
+        iCanvasTarget->UpdateOffScreenBitmapL(aForced);
+    }
+}
+
+// ---------------------------------------------------------------------------
+// From MMIDComponentNgaExtension
+// CMIDGraphics::FillEglSurface()
+// ---------------------------------------------------------------------------
+//
+TBool CMIDGraphics::FillEglSurface(const TRect& aRect, const TRgb& aColor)
+{
+    if (iCanvasTarget)
+    {
+        TRect rect(aRect);
+        rect.Move(iGraphics->CurrentOriginInTargetCoords());
+        rect.Intersection(iGraphics->CurrentClipRectInTargetCoords());
+        TSize canvasSize = iCanvasTarget->Control().Size();
+        rect.Intersection(TRect(iCanvasTarget->Control().Size()));
+        return iCanvasTarget->FillEglSurface(rect, aColor);
+    }
+    return EFalse;
+}
+
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::CanvasTargetSize()
+// ---------------------------------------------------------------------------
+//
+TSize CMIDGraphics::CanvasTargetSize() const
+{
+    if (iCanvasTarget)
+    {
+        return iCanvasTarget->ContentSize();
+    }
+    else
+    {
+        return TSize();
+    }
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::GetTargetCanvas()
+// ---------------------------------------------------------------------------
+//
+MMIDCanvas* CMIDGraphics::GetTargetCanvas() const
+{
+    return iCanvasTarget;
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::UpdateRect()
+// Merges aRect to member TRects that are used for tracking the updated canvas
+// area. Current coordinate translation and clip rect are taken into account.
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::UpdateRect(const TRect& aRect)
+{
+    // Translate rect to target (=canvas) coordinates
+    TRect rect(aRect);
+    rect.Move(iGraphics->CurrentOriginInTargetCoords());
+
+    // Clip with current clip rect and canvas bounds
+    rect.Intersection(iGraphics->CurrentClipRectInTargetCoords());
+
+    iCanvasTarget->UpdateRect(rect);
+}
+
+// ---------------------------------------------------------------------------
+// CMIDGraphics::UpdateTextRect()
+// Calculates bounding box for text.
+// ---------------------------------------------------------------------------
+//
+void CMIDGraphics::UpdateTextRect(TPtrC& aTextPtr, const TPoint& aP, TInt aAnchor)
+{
+    CFont* font = iGraphics->CurrentFont();
+
+    // Create a bounding box for the text
+    // TextWidthInPixels() seems to return few pixels too small value.
+    // Width of the zero char is added to be sure that bouding box is
+    // long enough.
+    TRect boundingBox(aP, TSize(font->TextWidthInPixels(aTextPtr) +
+                                font->WidthZeroInPixels(), 0));
+    const CAknLayoutFont* layoutFont = CAknLayoutFont::AsCAknLayoutFontOrNull(font);
+
+    if (layoutFont)
+    {
+        boundingBox.SetHeight(layoutFont->TextPaneHeight());
+    }
+    else
+    {
+        boundingBox.SetHeight(font->FontMaxHeight());
+    }
+
+    // Moving the bounding box according to the anchor. The EAnchorTop is
+    // default vertically and the EAnchorLeft is default horizontally
+    if (aAnchor & EAnchorBaseline)
+    {
+        boundingBox.Move(0, -font->FontMaxAscent());
+    }
+    else if (aAnchor & EAnchorBottom)
+    {
+        boundingBox.Move(0, -boundingBox.Height());
+    }
+
+    if (aAnchor & EAnchorHCenter)
+    {
+        boundingBox.Move(-(boundingBox.Width() / 2), 0);
+    }
+    else if (aAnchor & EAnchorRight)
+    {
+        boundingBox.Move(-boundingBox.Width(), 0);
+    }
+    UpdateRect(boundingBox);
+}
+
+#endif // RD_JAVA_NGA_ENABLED