1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/graphicsdeviceinterface/directgdi/src/directgdicontext.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,3588 @@
1.4 +// Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include "directgdicontext.h"
1.20 +#include "directgdipaniccodes.h"
1.21 +#include "directgdifont.h"
1.22 +#include "directgdidriver.h"
1.23 +#include <graphics/directgdidrawablesource.h>
1.24 +#include <graphics/directgdiengine.h>
1.25 +#include <e32cmn.h>
1.26 +#include <s32mem.h>
1.27 +#include <shapeinfo.h>
1.28 +#include <fbs.h>
1.29 +#include <fntstore.h>
1.30 +
1.31 +using namespace DirectGdi;
1.32 +
1.33 +/**
1.34 +CDirectGdiContext InternalizeL/ExternalizeL - version numbers.
1.35 +Add new version numbers here. A reason of adding new version numbers may be adding new
1.36 +CDirectGdiContext data members, which may have to be externalized/internalized.
1.37 +When that happens:
1.38 +1.Put a new enum item (like EDirectGDIContext_Ver01) with a version number, which is greater than
1.39 +the last version number, that was used.
1.40 +2.Document the new enum item.
1.41 +3.Update KDirectGDIContext_VerNo value to be the new enum item value.
1.42 +4.Update InternalizeL/ExternalizeL methods after adding the new version number.
1.43 +For example: If a new member is added to the class - TInt iSmth, when InternalizeL
1.44 +is called to operate on older archive, iSmth member won't be in the archive!
1.45 +So, in InternalizeL, there should be a check, something like:
1.46 +TUint16 archiveVerNo = 0;
1.47 + aReadStream >> archiveVerNo;
1.48 + if(archiveVerNo < EDirectGDIContext_Ver02) //EDirectGDIContext_Ver02 has been added, when iSmth has been added
1.49 + {
1.50 + //Do nothing - iSmth is not in the archive
1.51 + //Initialize it with some default value
1.52 + iSmth = KDefVal;
1.53 + }
1.54 + else
1.55 + {
1.56 + aReadStream >> iSmth;
1.57 + }
1.58 +*/
1.59 +enum
1.60 + {
1.61 + EDirectGDIContext_Ver01 = 1 //Base version number, when InternalizeL/ExternalizeL were added
1.62 + };
1.63 +
1.64 +LOCAL_D const TUint16 KDirectGDIContext_VerNo = EDirectGDIContext_Ver01;
1.65 +
1.66 +/**
1.67 +Static two-phase factory constuctor. Creates an object of this class.
1.68 +
1.69 +@param aDirectGdiDriver The driver object which provides an abstract factory for creating a concrete drawing engine.
1.70 +
1.71 +@pre CDirectGdiDriver object has been initialised from the calling thread.
1.72 +@post Instances of CDirectGdiContext and MDirectGdiEngine have been created.
1.73 +
1.74 +@leave Leaves if CDirectGdiDriver has not been initialised from the calling thread, or other errors occur during object construction.
1.75 +@return Reference to constructed CDirectGdiContext.
1.76 +*/
1.77 +EXPORT_C CDirectGdiContext* CDirectGdiContext::NewL(CDirectGdiDriver& aDirectGdiDriver)
1.78 + {
1.79 + CDirectGdiContext* result = new(ELeave) CDirectGdiContext(aDirectGdiDriver);
1.80 + CleanupStack::PushL(result);
1.81 +
1.82 + // Cache a reference to the driver locally, so that we do not need to use
1.83 + // the singleton accessor all the time. The singleton accessor utilises TLS
1.84 + // and hence incurs a performance penalty.
1.85 + //
1.86 + result->ConstructL();
1.87 +
1.88 + CleanupStack::Pop();
1.89 + return result;
1.90 + }
1.91 +
1.92 +
1.93 +/**
1.94 +Installs a rendering engine instance from the supplied driver, records
1.95 +a reference to the driver (for use by the destructor), and initialises it.
1.96 +
1.97 +@pre Invoked by NewL().
1.98 +@post Instance of MDirectGdiEngine has been created.
1.99 +
1.100 +@leave If CDirectGdiDriver::CreateEngine() returns an error code.
1.101 +*/
1.102 +void CDirectGdiContext::ConstructL()
1.103 + {
1.104 + User::LeaveIfError(iDriver.CreateEngine(iEngine));
1.105 + Reset(); // Initialise context and engine to default values.
1.106 + }
1.107 +
1.108 +
1.109 +/**
1.110 +Object constructor. Declared private for better encapsulation. A factory method is
1.111 +provided to instantiate this class. This class is not intended for derivation. Binary
1.112 +compatibility needs to be maintained.
1.113 +
1.114 +@param aDirectGdiDriver The driver that coordinates management of DirectGDI resources.
1.115 +
1.116 +@pre The driver must be initialised.
1.117 +@post None.
1.118 +*/
1.119 +EXPORT_C CDirectGdiContext::CDirectGdiContext(CDirectGdiDriver& aDirectGdiDriver) :
1.120 + iDriver(aDirectGdiDriver)
1.121 + {
1.122 + }
1.123 +
1.124 +
1.125 +/**
1.126 +Unbind current target from drawing context and mark drawing engine for deletion.
1.127 +
1.128 +@pre None.
1.129 +@post Rendering engine has had targets unbound and is marked for deletion.
1.130 +*/
1.131 +EXPORT_C CDirectGdiContext::~CDirectGdiContext()
1.132 + {
1.133 + // If an engine has been bound, destroy it through the driver.
1.134 + if (iEngine)
1.135 + {
1.136 + iDriver.DestroyEngine(iEngine);
1.137 + }
1.138 +
1.139 + CleanUpBrushPattern();
1.140 + iClippingRegion.Close();
1.141 + }
1.142 +
1.143 +
1.144 +/**
1.145 +Binds a rendering target to this drawing context.
1.146 +
1.147 +Subsequent rendering operations after calling this method will apply to the new rendering
1.148 +target. The clipping region will be reset to none, in other words drawing will be clipped to the
1.149 +full area of the new target by default, other context states such as pen or brush colour,
1.150 +font, etc. will remain unchanged.
1.151 +
1.152 +This operation could fail, DirectGDI clients should always check the return value when
1.153 +calling this method. The error state is not modified by this function.
1.154 +
1.155 +@param aTarget DirectGDI rendering target.
1.156 +
1.157 +@pre Rendering target has been fully constructed.
1.158 +@post The drawing context is bound to the new rendering target.
1.159 +
1.160 +@return KErrNone if successful, KErrNotSupported if target is incompatible with the drawing context,
1.161 + otherwise one of the system-wide error codes.
1.162 +*/
1.163 +EXPORT_C TInt CDirectGdiContext::Activate(RDirectGdiImageTarget& aTarget)
1.164 + {
1.165 + GRAPHICS_TRACE("CDirectGdiContext::Activate");
1.166 + TInt err = iEngine->Activate(aTarget);
1.167 +
1.168 + iActivated = EFalse;
1.169 + if (err == KErrNone)
1.170 + {
1.171 + iActivated = ETrue;
1.172 + }
1.173 + return err;
1.174 + }
1.175 +
1.176 +/**
1.177 +Disables AutoUpdateJustification state.
1.178 +
1.179 +@see CDirectGdiContext::SetJustifyAutoUpdate
1.180 +*/
1.181 +EXPORT_C void CDirectGdiContext::NoJustifyAutoUpdate()
1.182 + {
1.183 + GRAPHICS_TRACE("CDirectGdiContext::NoJustifyAutoUpdate");
1.184 + iAutoUpdateJustification = EFalse;
1.185 + }
1.186 +
1.187 +/**
1.188 +Enables AutoUpdateJustification state.
1.189 +During the text drawing, some text parameters which impact on positioning the text on the screen will be updated and applied
1.190 +*/
1.191 +EXPORT_C void CDirectGdiContext::SetJustifyAutoUpdate()
1.192 + {
1.193 + GRAPHICS_TRACE("CDirectGdiContext::SetJustifyAutoUpdate");
1.194 + iAutoUpdateJustification = ETrue;
1.195 + }
1.196 +
1.197 +/**
1.198 +Draws the whole of a CFbsBitmap. Calls BitBlt() with a rectangle set to the extents of aBitmap.
1.199 +
1.200 +No scaling or stretching is involved. The current clipping region and drawing mode apply. The
1.201 +source bitmap can reside in system memory or ROM. Bitmaps in system memory may be compressed
1.202 +using RLE or Palette compression, and can be in any supported display mode. Please refer to
1.203 +CFbsBitmap documentation for more details.
1.204 +
1.205 +If the client modifies the content of the bitmap after issuing a BitBlt() command, the method
1.206 +does not guarantee whether the old bitmap content or the new one will be drawn. Clients must call
1.207 +Finish() on the driver before modifying the bitmap content if they want a guaranteed result that
1.208 +the previously issued BitBlt() will draw the old bitmap content.
1.209 +
1.210 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.211 +
1.212 +@see void CDirectGdiContext::BitBlt(const TPoint& aPoint, const CFbsBitmap* aBitmap, const TRect& aSourceRect);
1.213 +@param aDestPos The position to draw the top left corner of the bitmap.
1.214 +@param aSourceBitmap The source bitmap.
1.215 +
1.216 +@pre The rendering target has been activated.
1.217 +@post Request to draw the bitmap content has been accepted. There is no guarantee that the request
1.218 + has been processed when the method returns.
1.219 +
1.220 +@panic DGDI 7, if the rendering context has not been activated.
1.221 +*/
1.222 +EXPORT_C void CDirectGdiContext::BitBlt(const TPoint& aDestPos, const CFbsBitmap& aSourceBitmap)
1.223 + {
1.224 + GRAPHICS_TRACE2("CDirectGdiContext::BitBlt(%d,%d)", aDestPos.iX, aDestPos.iY);
1.225 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.226 +
1.227 + if (ValidateBitmap (aSourceBitmap))
1.228 + {
1.229 + iEngine->BitBlt(aDestPos, aSourceBitmap, TRect(aSourceBitmap.SizeInPixels()));
1.230 + }
1.231 + }
1.232 +
1.233 +
1.234 +/**
1.235 +Draws a particular rectangle from a CFbsBitmap via bitmap block image transfer.
1.236 +The bitmap content is specified by the given source rectangle. The bitmap content is drawn
1.237 +into the rendering target starting from the given destination position. To draw the content
1.238 +of the entire bitmap, a source rectangle TRect(TPoint(0,0), aBitmap.SizeInPixels()) is used.
1.239 +
1.240 +The source rectangle is intersected with the source bitmap’s full extent, and the intersection
1.241 +will become the effective value of the source rectangle.
1.242 +
1.243 +No scaling or stretching is involved. The current clipping region and drawing mode apply. The
1.244 +source bitmap can reside in system memory or ROM. Bitmaps in system memory may be compressed
1.245 +using RLE or Palette compression, and can be in any supported display mode. Please refer to
1.246 +CFbsBitmap documentation for more details.
1.247 +
1.248 +The bitmap is not guaranteed to continue to exist outside the implementation of this method, for example
1.249 +DirectGDI clients may delete the bitmap immediately after issuing a BitBlt() command. The adaptation
1.250 +must maintain its own copy by duplicating the bitmap if access to the bitmap data is required
1.251 +later on, for example outside this method implementation. Duplicating the bitmap will increase its
1.252 +reference counter and will prevent the object from being destroyed by Fbserv. The adaptation
1.253 +must make sure the bitmap copy is destroyed when it is no longer needed.
1.254 +
1.255 +If the client modifies the content of the bitmap after issuing BitBlt() command, the method
1.256 +does not guarantee whether the old bitmap content or the new one will be drawn. Clients must call
1.257 +Finish() on the driver before modifying the bitmap content if they want a guaranteed result that
1.258 +the previously issued BitBlt() will draw the old bitmap content.
1.259 +
1.260 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.261 +
1.262 +@param aDestPos The position to draw the top left corner of the bitmap (destination position).
1.263 +@param aSourceBitmap The source bitmap.
1.264 +@param aSourceRect A rectangle defining the piece of the source to be drawn. No image data
1.265 + is transferred if no intersection exists with the bitmap.
1.266 + NOTE: The top and left hand edges are inclusive, the bottom and
1.267 + right hand edge are exclusive.
1.268 +
1.269 +@pre The rendering target has been activated.
1.270 +@post Request to draw the bitmap content has been accepted. There is no guarantee that the request
1.271 + has been processed when the method returns.
1.272 +
1.273 +@panic DGDI 7, if the rendering context has not been activated.
1.274 +*/
1.275 +EXPORT_C void CDirectGdiContext::BitBlt(
1.276 + const TPoint& aDestPos,
1.277 + const CFbsBitmap& aSourceBitmap,
1.278 + const TRect& aSourceRect)
1.279 + {
1.280 + GRAPHICS_TRACE2("CDirectGdiContext::BitBlt(%d,%d)", aDestPos.iX, aDestPos.iY);
1.281 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.282 +
1.283 + if (ValidateBitmap(aSourceBitmap))
1.284 + {
1.285 + TRect sourceRect = IntersectBitmapWithRect(aSourceBitmap, aSourceRect);
1.286 + if (!sourceRect.IsEmpty())
1.287 + {
1.288 + iEngine->BitBlt(aDestPos, aSourceBitmap, sourceRect);
1.289 + }
1.290 + }
1.291 + }
1.292 +
1.293 +/**
1.294 +Validates a specified bitmap image.
1.295 +The bitmap is deemed invalid if its associated handle equals KNullHandle or it
1.296 +has a width or height of zero.
1.297 +
1.298 +The following errors are set in iDirectGdiDriver if the associated conditions are met:
1.299 + - KErrBadHandle if the handle associated with the bitmap equals KNullHandle.
1.300 + - KErrArgument is the bitmaps width or height is zero.
1.301 +
1.302 +@param aBitmap The bitmap to validate.
1.303 +
1.304 +@return ETrue if the specified bitmap is valid, otherwise EFalse.
1.305 +*/
1.306 +TBool CDirectGdiContext::ValidateBitmap(const CFbsBitmap& aBitmap)
1.307 + {
1.308 + TInt errorCode = KErrNone;
1.309 + TBool result = ETrue;
1.310 +
1.311 + if (aBitmap.Handle() == KNullHandle)
1.312 + {
1.313 + errorCode = KErrBadHandle;
1.314 + }
1.315 + else
1.316 + {
1.317 + // Check that mask and source bitmap have width and height.
1.318 + const TSize bitmapSize = aBitmap.SizeInPixels();
1.319 + if (!bitmapSize.iWidth || !bitmapSize.iHeight)
1.320 + {
1.321 + errorCode = KErrArgument;
1.322 + }
1.323 + }
1.324 +
1.325 + if (errorCode != KErrNone)
1.326 + {
1.327 + iDriver.SetError(errorCode);
1.328 + result = EFalse;
1.329 + }
1.330 +
1.331 + return result;
1.332 + }
1.333 +
1.334 +/*
1.335 +Returns a TRect that intersects both the CFbsBitmap and the passed-in TRect.
1.336 +
1.337 +@param aBitmap The bitmap to intersect with.
1.338 +@param aRect The TRect object that is overlapping the bitmap.
1.339 +@return A TRect that represents the intersection of the two. If the two do not intersect,
1.340 + it will be an empty TRect object.
1.341 +*/
1.342 +TRect CDirectGdiContext::IntersectBitmapWithRect(const CFbsBitmap& aBitmap, const TRect& aRect) const
1.343 + {
1.344 + TRect result = TRect(aBitmap.SizeInPixels());
1.345 +
1.346 + if (aRect == result)
1.347 + return result;
1.348 +
1.349 + if (result.Intersects(aRect))
1.350 + {
1.351 + result.Intersection(aRect);
1.352 + return result;
1.353 + }
1.354 +
1.355 + return TRect(0,0,0,0);
1.356 + }
1.357 +
1.358 +/**
1.359 +
1.360 +Validates the specified source bitmap and mask pair.
1.361 +First validates the source and mask bitmaps individually, and then checks for valid
1.362 +bitmap pairings. The only pairing not supported is: EGray256 masks and sources with
1.363 +an alpha-channel when using EDrawModeWriteAlpha.
1.364 +
1.365 +@see CDirectGdiContext::ValidateBitmap()
1.366 +
1.367 +The following errors are set in iDirectGdiDriver if the associated conditions are met:
1.368 + - KErrBadHandle if the handle associated with either bitmap equals KNullHandle.
1.369 + - KErrArgument if either bitmaps width or height is zero.
1.370 + - KErrArgument if the mask format is EGray256 and the source contains an alpha-channel and draw mode is EDrawModeWriteAlpha.
1.371 +
1.372 +@param aSourceBitmap The source bitmap to validate.
1.373 +@param aMaskBitmap The mask to validate.
1.374 +
1.375 +@return ETrue if the specified bitmaps is valid, otherwise EFalse.
1.376 +*/
1.377 +TBool CDirectGdiContext::ValidateSourceAndMaskBitmaps(const CFbsBitmap& aSourceBitmap, const CFbsBitmap& aMaskBitmap)
1.378 + {
1.379 + TInt errorCode = KErrNone;
1.380 + TBool result = ETrue;
1.381 +
1.382 + if (ValidateBitmap (aSourceBitmap) && ValidateBitmap (aMaskBitmap))
1.383 + {
1.384 + TDisplayMode sourceBitmapDisplayMode = aSourceBitmap.DisplayMode();
1.385 +
1.386 + // We do not currently support EGray256 masks and sources with an alpha-channel
1.387 + // with EDrawModeWriteAlpha.
1.388 + if ((iDrawMode == DirectGdi::EDrawModeWriteAlpha) &&
1.389 + (aMaskBitmap.DisplayMode() == EGray256) &&
1.390 + ((sourceBitmapDisplayMode == EColor16MA) || (sourceBitmapDisplayMode == EColor16MAP)))
1.391 + {
1.392 + errorCode = KErrArgument;
1.393 + }
1.394 + }
1.395 + else
1.396 + {
1.397 + result = EFalse;
1.398 + }
1.399 +
1.400 + if (errorCode != KErrNone)
1.401 + {
1.402 + iDriver.SetError(errorCode);
1.403 + result = EFalse;
1.404 + }
1.405 +
1.406 + return result;
1.407 + }
1.408 +
1.409 +/**
1.410 +Helper method for performing BitBltMasked.
1.411 +Note that aInvertMask is ignored if aMaskPos is not at 0,0.
1.412 +
1.413 +@see CDirectGdiContext::BitBlt(const TPoint& aPoint, const CFbsBitmap& aBitmap, const TRect& aSourceRect);
1.414 +@see CDirectGdiContext::BitBltMasked(const TPoint&, const CFbsBitmap&,const TRect&, const CFbsBitmap&, const TPoint&);
1.415 +
1.416 +@param aDestPos The destination for the top left corner of the transferred bitmap.
1.417 + It is relative to the top left corner of the destination bitmap, which may be the screen.
1.418 +@param aSourceBitmap A memory-resident source bitmap.
1.419 +@param aSourceRect A rectangle defining the piece of the bitmap to be drawn,
1.420 + with co-ordinates relative to the top left corner of the bitmap.
1.421 +@param aMaskBitmap Mask bitmap.
1.422 +@param aInvertMask If EFalse, a source pixel that is masked by a black pixel is not transferred to
1.423 + the destination rectangle. If ETrue, then a source pixel that is masked by a
1.424 + white pixel is not transferred to the destination rectangle. If alpha blending
1.425 + is used instead of masking, this flag is ignored and no inversion takes place.
1.426 + Note that this parameter is ignored if aMaskPos does not equal TPoint(0,0).
1.427 +@param aMaskPos The point on the mask bitmap to use as the top left corner
1.428 +*/
1.429 +void CDirectGdiContext::DoBitBltMasked(
1.430 + const TPoint& aDestPos,
1.431 + const CFbsBitmap& aSourceBitmap,
1.432 + const TRect& aSourceRect,
1.433 + const CFbsBitmap& aMaskBitmap,
1.434 + TBool aInvertMask,
1.435 + const TPoint& aMaskPos)
1.436 + {
1.437 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.438 +
1.439 + if (ValidateSourceAndMaskBitmaps(aSourceBitmap, aMaskBitmap))
1.440 + {
1.441 + // If the source rectangle does not intersect aBitmap, do nothing.
1.442 + TRect sourceRect = IntersectBitmapWithRect(aSourceBitmap, aSourceRect);
1.443 + if (!sourceRect.IsEmpty())
1.444 + {
1.445 + if (aMaskPos == TPoint(0, 0))
1.446 + {
1.447 + iEngine->BitBltMasked(aDestPos, aSourceBitmap, sourceRect, aMaskBitmap, aInvertMask);
1.448 + }
1.449 + else
1.450 + {
1.451 + TSize maskSize = aMaskBitmap.SizeInPixels();
1.452 + // Convert negative or large mask offsets into sensible positive ones for tiling.
1.453 + TPoint maskOffset(aMaskPos.iX % maskSize.iWidth, aMaskPos.iY % maskSize.iHeight);
1.454 + if (maskOffset.iX < 0)
1.455 + maskOffset.iX += maskSize.iWidth;
1.456 + if (maskOffset.iY < 0)
1.457 + maskOffset.iY += maskSize.iHeight;
1.458 +
1.459 + iEngine->BitBltMasked(aDestPos, aSourceBitmap, sourceRect, aMaskBitmap, maskOffset);
1.460 + }
1.461 + }
1.462 + }
1.463 + }
1.464 +
1.465 +
1.466 +/**
1.467 +Performs a masked bitmap block transfer. Source rectangle operates in a similar way to BitBlt().
1.468 +
1.469 +This function uses either a black and white (binary) mask bitmap, or if the mask's display mode is
1.470 +EGray256, alpha blending is used. The result is undefined if the mask pixel value is neither black nor
1.471 +white and the mask display mode is other than EGray256.
1.472 +
1.473 +The mask is aligned with the source bitmap by aligning the first pixel of the mask and source bitmaps within
1.474 +the source rectangle. Tiling in both directions applies if the mask size is smaller than the source rectangle.
1.475 +Note that the mask is applied before the piece of the bitmap is defined - the mask is tiled relative to the
1.476 +top left of the original source bitmap rather than the top left of the bitmap piece. If the mask has zero
1.477 +width or height, the error state is set to KErrArgument and no drawing is performed.
1.478 +
1.479 +The mask bitmap can be used as either a positive or negative mask. Masked pixels are not mapped to the
1.480 +destination rectangle.
1.481 +
1.482 +If the client modifies the contents of the bitmap or the mask after issuing the BitBltMasked() command, the
1.483 +method does not guarantee whether the old bitmap or mask contents, or the new ones will be used.
1.484 +Clients must call Finish() on the driver before modifying the bitmap or mask contents if they want a
1.485 +guaranteed result that the previously issued BitBltMasked() will be using the old bitmap or mask contents.
1.486 +
1.487 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.488 +
1.489 +@see CDirectGdiContext::BitBlt(const TPoint& aPoint, const CFbsBitmap& aBitmap, const TRect& aSourceRect);
1.490 +@see CDirectGdiContext::BitBltMasked(const TPoint&, const CFbsBitmap&,const TRect&, const CFbsBitmap&, const TPoint&);
1.491 +
1.492 +@param aDestPos The destination for the top left corner of the transferred bitmap.
1.493 + It is relative to the top left corner of the destination bitmap, which may be the screen.
1.494 +@param aSourceBitmap A memory-resident source bitmap.
1.495 +@param aSourceRect A rectangle defining the piece of the bitmap to be drawn,
1.496 + with co-ordinates relative to the top left corner of the bitmap.
1.497 +@param aMaskBitmap Mask bitmap.
1.498 +@param aInvertMask If EFalse, a source pixel that is masked by a black pixel is not transferred to
1.499 + the destination rectangle. If ETrue, then a source pixel that is masked by a
1.500 + white pixel is not transferred to the destination rectangle. If alpha blending
1.501 + is used instead of masking, this flag is ignored and no inversion takes place.
1.502 +
1.503 +@pre The rendering target has been activated. aBitmap and aMask hold valid handles.
1.504 +@post Request to draw the masked bitmap content has been accepted.
1.505 + There is no guarantee that the request has been processed when the method returns.
1.506 +
1.507 +@panic DGDI 7, if the rendering context has not been activated.
1.508 +*/
1.509 +EXPORT_C void CDirectGdiContext::BitBltMasked(
1.510 + const TPoint& aDestPos,
1.511 + const CFbsBitmap& aSourceBitmap,
1.512 + const TRect& aSourceRect,
1.513 + const CFbsBitmap& aMaskBitmap,
1.514 + TBool aInvertMask)
1.515 + {
1.516 + GRAPHICS_TRACE2("CDirectGdiContext::BitBltMasked(%d,%d)", aDestPos.iX, aDestPos.iY);
1.517 + DoBitBltMasked (aDestPos, aSourceBitmap, aSourceRect, aMaskBitmap, aInvertMask, TPoint(0, 0));
1.518 + }
1.519 +
1.520 +
1.521 +/**
1.522 +Performs a masked bitmap block transfer. Source rectangle operates in a similar way to BitBlt().
1.523 +
1.524 +This function uses either a black and white (binary) mask bitmap, or if aMaskBitmap's display mode is
1.525 +EGray256, alpha blending is used. The result is undefined if the mask pixel value is neither black nor
1.526 +white and the mask display mode is other than EGray256.
1.527 +
1.528 +The source rectangle is intersected with the source bitmap’s full extent and the intersection
1.529 +will become the effective value. The mask bitmap is aligned with the source bitmap by aligning
1.530 +its pixel at the position specified in aMaskPt with the first pixel of the source bitmap
1.531 +within the source rectangle. The mask bitmap will be tiled if it is smaller than the source
1.532 +rectangle. If the mask has zero width or height, the error state is set to KErrArgument and
1.533 +no drawing is performed.
1.534 +
1.535 +If the client modifies the contents of the bitmap or the mask after issuing the BitBltMasked() command, the
1.536 +method does not guarantee whether the old bitmap or mask contents, or the new ones will be used.
1.537 +Clients must call Finish() on the driver before modifying the bitmap or mask contents if they want a
1.538 +guaranteed result that the previously issued BitBltMasked() will be using the old bitmap or mask contents.
1.539 +
1.540 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.541 +
1.542 +@see CDirectGdiContext::BitBlt(const TPoint& aPoint, const CFbsBitmap& aBitmap, const TRect& aSourceRect);
1.543 +@see CDirectGdiContext::BitBltMasked(const TPoint&, const CFbsBitmap&,const TRect&, const CFbsBitmap&, TBool);
1.544 +
1.545 +@param aDestPos The destination for the top left corner of the transferred bitmap.
1.546 + It is relative to the top left corner of the destination bitmap, which may be the screen.
1.547 +@param aSourceBitmap A memory-resident source bitmap.
1.548 +@param aSourceRect A rectangle defining the piece of the bitmap to be drawn,
1.549 + with co-ordinates relative to the top left corner of the bitmap.
1.550 +@param aMaskBitmap Mask bitmap.
1.551 +@param aMaskPos The point on the mask bitmap to use as the top left corner
1.552 +
1.553 +@pre The rendering target has been activated. aBitmap and aMask hold valid handles.
1.554 +@post Request to draw the masked bitmap content has been accepted.
1.555 + There is no guarantee that the request has been processed when the method returns.
1.556 +
1.557 +@panic DGDI 7, if no context is activated
1.558 +*/
1.559 +EXPORT_C void CDirectGdiContext::BitBltMasked(
1.560 + const TPoint& aDestPos,
1.561 + const CFbsBitmap& aSourceBitmap,
1.562 + const TRect& aSourceRect,
1.563 + const CFbsBitmap& aMaskBitmap,
1.564 + const TPoint& aMaskPos)
1.565 + {
1.566 + GRAPHICS_TRACE2("CDirectGdiContext::BitBltMasked(%d,%d)", aDestPos.iX, aDestPos.iY);
1.567 + DoBitBltMasked (aDestPos, aSourceBitmap, aSourceRect, aMaskBitmap, EFalse, aMaskPos);
1.568 + }
1.569 +
1.570 +
1.571 +/**
1.572 +Resets the current clipping region to none.
1.573 +
1.574 +@see CDirectGdiContext::SetClippingRegion(const TRegion&)
1.575 +
1.576 +@pre The rendering target has been activated.
1.577 +@post Clipping region is reset to none. Subsequent rendering operations on current target will be clipped to the
1.578 + full area of the target.
1.579 +
1.580 +@panic DGDI 7, if the rendering context has not been activated.
1.581 +*/
1.582 +EXPORT_C void CDirectGdiContext::ResetClippingRegion()
1.583 + {
1.584 + GRAPHICS_TRACE("CDirectGdiContext::ResetClippingRegion");
1.585 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.586 + iClippingRegion.Clear();
1.587 + iEngine->ResetClippingRegion();
1.588 + }
1.589 +
1.590 +
1.591 +/**
1.592 +Clears the entire target area with the current brush colour. The area is filled
1.593 +as if ESolidBrush is used. Current clipping region and drawing mode apply.
1.594 +
1.595 +@see CDirectGdiContext::Clear(const TRect&)
1.596 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.597 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.598 +@see CDirectGdiContext::SetClippingRegion(const TRegion& aRegion)
1.599 +
1.600 +@pre The rendering target has been activated.
1.601 +@post Request to clear given rectangular area (clipped to current clipping region)
1.602 + has been accepted. There is no guarantee that the request has been processed
1.603 + when the method returns.
1.604 +
1.605 +@panic DGDI 7, if the rendering context has not been activated.
1.606 +*/
1.607 +EXPORT_C void CDirectGdiContext::Clear()
1.608 + {
1.609 + GRAPHICS_TRACE("CDirectGdiContext::Clear");
1.610 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.611 + iEngine->Clear();
1.612 + }
1.613 +
1.614 +
1.615 +/**
1.616 +Clears the given rectangular area with current brush colour. The area is filled
1.617 +as if ESolidBrush is used. Current clipping region and drawing mode apply.
1.618 +
1.619 +@see CDirectGdiContext::Clear()
1.620 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.621 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.622 +@see CDirectGdiContext::SetClippingRegion(const TRegion&)
1.623 +
1.624 +@param aRect Area to be cleared.
1.625 +
1.626 +@pre The rendering target has been activated.
1.627 +@post Request to clear given rectangular area (clipped to current clipping region) has been accepted.
1.628 + There is no guarantee that the request has been processed when the method returns.
1.629 +
1.630 +@panic DGDI 7, if the rendering context has not been activated.
1.631 +*/
1.632 +EXPORT_C void CDirectGdiContext::Clear(const TRect& aRect)
1.633 + {
1.634 + GRAPHICS_TRACE2("CDirectGdiContext::Clear(%d,%d)", aRect.Width(), aRect.Height());
1.635 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.636 + if(aRect.Width() <= 0 || aRect.Height() <= 0)
1.637 + {
1.638 + return;
1.639 + }
1.640 + iEngine->Clear(aRect);
1.641 + }
1.642 +
1.643 +
1.644 +/**
1.645 +Resets the bitmap brush pattern to none. If the current brush style is EPatternedBrush, the brush
1.646 +style will be set to ENullBrush.
1.647 +
1.648 +@see CDirectGdiContext::SetBrushPattern(const CFbsBitmap&)
1.649 +
1.650 +@pre None.
1.651 +@post The bitmap brush pattern is no longer used.
1.652 +*/
1.653 +EXPORT_C void CDirectGdiContext::ResetBrushPattern()
1.654 + {
1.655 + GRAPHICS_TRACE("CDirectGdiContext::ResetBrushPattern");
1.656 + CleanUpBrushPattern();
1.657 + if (iBrushStyle == DirectGdi::EPatternedBrush)
1.658 + {
1.659 + iBrushStyle = DirectGdi::ENullBrush;
1.660 + iEngine->SetBrushStyle(iBrushStyle);
1.661 + }
1.662 + iEngine->ResetBrushPattern();
1.663 + }
1.664 +
1.665 +
1.666 +/**
1.667 +Releases the selected font for this context.
1.668 +
1.669 +@pre None.
1.670 +@post Internal resources previously allocated during SetFont() are released.
1.671 +
1.672 +@see CDirectGdiContext::SetFont()
1.673 +@see CGraphicsContext::DiscardFont()
1.674 +*/
1.675 +EXPORT_C void CDirectGdiContext::ResetFont()
1.676 + {
1.677 + GRAPHICS_TRACE("CDirectGdiContext::ResetFont");
1.678 + iEngine->ResetFont();
1.679 + iFont.Reset();
1.680 + }
1.681 +
1.682 +
1.683 +/**
1.684 +Draws an arc. An arc is a segment of an ellipse which is defined by a given rectangle. The arc is drawn
1.685 +anti-clockwise from the arc start point to the arc end point. The arc start point is the intersection
1.686 +between vectors from the centre of the ellipse to the given start position and the ellipse.
1.687 +Arc end point is defined in the same way.
1.688 +
1.689 +@param aRect The rectangle which defines where to draw the ellipse.
1.690 +@param aStart Position to be used in defining the arc start point.
1.691 +@param aEnd Position to be used in defining the arc end point.
1.692 +
1.693 +@pre The rendering target has been activated.
1.694 +@post Request to draw an arc has been accepted. There is no guarantee that the request
1.695 + has been processed when the method returns.
1.696 +
1.697 +@panic DGDI 7, if the rendering context has not been activated.
1.698 +
1.699 +@see CDirectGdiContext::DrawPie(const TRect&, const TPoint&, const TPoint&)
1.700 +*/
1.701 +EXPORT_C void CDirectGdiContext::DrawArc(const TRect& aRect, const TPoint& aStart, const TPoint& aEnd)
1.702 + {
1.703 + GRAPHICS_TRACE("CDirectGdiContext::DrawArc");
1.704 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.705 + if (aRect.IsEmpty() || iPenStyle == DirectGdi::ENullPen || (iPenSize.iWidth == 0) || (iPenSize.iHeight == 0))
1.706 + {
1.707 + return;
1.708 + }
1.709 + iEngine->DrawArc(aRect, aStart, aEnd);
1.710 + }
1.711 +
1.712 +
1.713 +/**
1.714 +Draws and fills a pie. A pie is a shape defined by an arc from the ellipse and straight lines
1.715 +from the centre of the ellipse to the arc start and end position.
1.716 +
1.717 +@param aRect The rectangle which defines where to draw the ellipse
1.718 +@param aStart Position to be used in defining the arc start point
1.719 +@param aEnd Position to be used in defining the arc end point
1.720 +
1.721 +@pre The rendering target has been activated.
1.722 +@post Request to draw a pie has been accepted. There is no guarantee that the request
1.723 + has been processed when the method returns.
1.724 +
1.725 +@panic DGDI 7, if the rendering context has not been activated.
1.726 +@panic DGDI 9, if the brush style is EPatternedBrush but no pattern has been set.
1.727 +
1.728 +@see CDirectGdiContext::DrawArc(const TRect&, const TPoint&, const TPoint&)
1.729 +*/
1.730 +EXPORT_C void CDirectGdiContext::DrawPie(const TRect& aRect, const TPoint& aStart, const TPoint& aEnd)
1.731 + {
1.732 + GRAPHICS_TRACE("CDirectGdiContext::DrawPie");
1.733 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.734 + GRAPHICS_ASSERT_ALWAYS(iBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.735 +
1.736 + if (aRect.IsEmpty())
1.737 + {
1.738 + return;
1.739 + }
1.740 + iEngine->DrawPie(aRect, aStart, aEnd);
1.741 + }
1.742 +
1.743 +/**
1.744 +Draws the bitmap contents into the destination rectangle. Scaling applies when
1.745 +the extents of the source bitmap and the destination rectangle do not match.
1.746 +If the source rectangle is not completely contained within the source
1.747 +bitmap's extents, no drawing will take place.
1.748 +
1.749 +If the client modifies the content of the bitmap after issuing a DrawBitmap() command,
1.750 +the method does not guarantee that the old bitmap content or the new one
1.751 +will be drawn. Clients must call Finish() on the driver before modifying the bitmap
1.752 +content if they want a guarantee that the previously issued DrawBitmap() will draw the
1.753 +old bitmap content.
1.754 +
1.755 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.756 +
1.757 +@see CDirectGdiContext::DrawBitmap(const TRect&, const CFbsBitmap&, const TRect&)
1.758 +
1.759 +@param aDestRect Destination rectangle.
1.760 +@param aSourceBitmap Source bitmap.
1.761 +
1.762 +@pre The rendering target has been activated.
1.763 +@post Request to draw the bitmap content has been accepted.
1.764 + There is no guarantee that the request has been processed when the method returns.
1.765 +
1.766 +@panic DGDI 7, if the rendering context has not been activated.
1.767 +*/
1.768 +EXPORT_C void CDirectGdiContext::DrawBitmap(const TRect& aDestRect, const CFbsBitmap& aSourceBitmap)
1.769 + {
1.770 + GRAPHICS_TRACE2("CDirectGdiContext::DrawBitmap(%d,%d)", aDestRect.iTl.iX, aDestRect.iTl.iY);
1.771 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.772 + if (ValidateBitmap (aSourceBitmap))
1.773 + {
1.774 + DrawBitmap(aDestRect, aSourceBitmap,TRect(TPoint(0, 0), aSourceBitmap.SizeInPixels()));
1.775 + }
1.776 + }
1.777 +
1.778 +/**
1.779 +Draws the bitmap contents into the destination rectangle. Scaling applies when
1.780 +the destination and source rectangles do not match. The source bitmap may be
1.781 +compressed. The destination rectangle will be clipped with the current clipping
1.782 +region. If the source rectangle is not completely contained within the source
1.783 +bitmap's extents, no drawing will take place.
1.784 +
1.785 +If the client modifies the content of the bitmap after issuing a DrawBitmap() command,
1.786 +the method does not guarantee that the old bitmap content or the new one
1.787 +will be drawn. Clients must call Finish() on the driver before modifying the bitmap
1.788 +content if they want a guarantee that the previously issued DrawBitmap() will draw the
1.789 +old bitmap content.
1.790 +
1.791 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.792 +
1.793 +@see CDirectGdiContext::DrawBitmap(const TRect&, const CFbsBitmap&)
1.794 +
1.795 +@param aDestRect Destination rectangle.
1.796 +@param aSourceBitmap Source bitmap.
1.797 +@param aSourceRect Rectangle specifying the area of the source bitmap to be drawn.
1.798 +
1.799 +@pre The rendering target has been activated.
1.800 +@post Request to draw the bitmap content has been accepted.
1.801 + There is no guarantee that the request has been processed when the method returns.
1.802 +
1.803 +@panic DGDI 7, if the rendering context has not been activated.
1.804 +*/
1.805 +EXPORT_C void CDirectGdiContext::DrawBitmap(const TRect& aDestRect, const CFbsBitmap& aSourceBitmap, const TRect& aSourceRect)
1.806 + {
1.807 + GRAPHICS_TRACE2("CDirectGdiContext::DrawBitmap(%d,%d)", aDestRect.iTl.iX, aDestRect.iTl.iY);
1.808 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.809 +
1.810 + if (ValidateBitmap(aSourceBitmap))
1.811 + {
1.812 + TSize sourceSize = aSourceBitmap.SizeInPixels();
1.813 + // If source rectangle is not fully contained by the extents of the source bitmap,
1.814 + // or the size of the source bitmap is zero, do nothing.
1.815 + if (aSourceRect.iTl.iX >= 0 &&
1.816 + aSourceRect.iTl.iY >= 0 &&
1.817 + aSourceRect.iBr.iX <= sourceSize.iWidth &&
1.818 + aSourceRect.iBr.iY <= sourceSize.iHeight &&
1.819 + !aDestRect.IsEmpty() && !aSourceRect.IsEmpty())
1.820 + {
1.821 + iEngine->DrawBitmap(aDestRect, aSourceBitmap, aSourceRect);
1.822 + }
1.823 + }
1.824 + }
1.825 +
1.826 +
1.827 +/**
1.828 +Draws bitmap content with masking. Scaling applies to both the source bitmap and the mask.
1.829 +Both the source and the mask bitmap may be compressed. The destination and source rectangles
1.830 +operate in a similar way to DrawBitmap().
1.831 +
1.832 +This function uses either a black and white (binary) mask bitmap, or if the mask's display mode is
1.833 +EGray256, alpha blending is used. The result is undefined if the mask pixel value is neither black nor
1.834 +white and the mask display mode is other than EGray256.
1.835 +
1.836 +The mask is aligned with the source bitmap by aligning their first pixels within the source
1.837 +rectangle. If the mask size is greater than or equal to the source bitmap size, it will be
1.838 +scaled to fit the destination rectangle in the same way the source bitmap is scaled.
1.839 +If the mask has zero width or height, the error state is set to KErrArgument and no drawing is performed.
1.840 +
1.841 +If the mask is smaller than the source bitmap, it will be tiled to fit the source bitmap size,
1.842 +and then scaled to fit the destination rectangle.
1.843 +
1.844 +If the client modifies the content of the bitmap or mask after issuing a DrawBitmapMasked() command,
1.845 +the method does not guarantee whether the old bitmap or mask contents, or the new ones
1.846 +will be used. Clients must call Finish() on the driver before modifying the bitmap or mask contents
1.847 +if they want a guaranteed result that the previously issued DrawBitmapMasked() will be using the old
1.848 +bitmap or mask contents.
1.849 +
1.850 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.851 +
1.852 +@param aDestRect Destination rectangle.
1.853 +@param aSourceBitmap Source bitmap.
1.854 +@param aSourceRect Rectangle specifying the area of the source bitmap to be drawn.
1.855 +@param aMaskBitmap Mask bitmap.
1.856 +@param aInvertMask If EFalse, a source pixel that is masked by a black pixel is not transferred to
1.857 + the destination rectangle. If ETrue, then a source pixel that is masked by a
1.858 + white pixel is not transferred to the destination rectangle. If alpha blending
1.859 + is used instead of masking, this flag is ignored and no inversion takes place.
1.860 +
1.861 +@pre The rendering target has been activated.
1.862 +@post Request to draw the bitmap content with masking has been accepted.
1.863 + There is no guarantee that the request has been processed when the method returns.
1.864 +
1.865 +@panic DGDI 7, if the rendering context has not been activated.
1.866 +*/
1.867 +EXPORT_C void CDirectGdiContext::DrawBitmapMasked(
1.868 + const TRect& aDestRect,
1.869 + const CFbsBitmap& aSourceBitmap,
1.870 + const TRect& aSourceRect,
1.871 + const CFbsBitmap& aMaskBitmap,
1.872 + TBool aInvertMask)
1.873 + {
1.874 + GRAPHICS_TRACE2("CDirectGdiContext::DrawBitmapMasked(%d,%d)", aDestRect.iTl.iX, aDestRect.iTl.iY);
1.875 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.876 +
1.877 + if (ValidateSourceAndMaskBitmaps(aSourceBitmap, aMaskBitmap))
1.878 + {
1.879 + TSize sourceSize = aSourceBitmap.SizeInPixels();
1.880 + // Ensure source rect is fully within bounds of bitmap extents and
1.881 + // dest and source rect are not empty.
1.882 + if (aSourceRect.iTl.iX >= 0 &&
1.883 + aSourceRect.iTl.iY >= 0 &&
1.884 + aSourceRect.iBr.iX <= sourceSize.iWidth &&
1.885 + aSourceRect.iBr.iY <= sourceSize.iHeight &&
1.886 + !aDestRect.IsEmpty() && !aSourceRect.IsEmpty())
1.887 + {
1.888 + iEngine->DrawBitmapMasked(aDestRect, aSourceBitmap, aSourceRect, aMaskBitmap, aInvertMask);
1.889 + }
1.890 + }
1.891 + }
1.892 +
1.893 +/**
1.894 +Draws and fills a rectangle with rounded corners. The corner is constructed as an arc of
1.895 +an ellipse. The outline is drawn in the current pen colour, size and style if the pen
1.896 +colour is not ENullPen. The area inside the rectangle is filled according to the current
1.897 +brush colour and style.
1.898 +
1.899 +If the corner size has zero width or height, a square-cornered rectangle is drawn. If the corner
1.900 +size is greater than half the extents of the rectangle, an ellipse is drawn.
1.901 +
1.902 +@param aRect The rectangle.
1.903 +@param aCornerSize The corner size.
1.904 +
1.905 +@see CDirectGdiContext::SetPenSize()
1.906 +@see CDirectGdiContext::SetPenStyle()
1.907 +@see CDirectGdiContext::SetPenColor()
1.908 +@see CDirectGdiContext::SetBrushColor()
1.909 +@see CDirectGdiContext::SetBrushStyle()
1.910 +@see CDirectGdiContext::SetBrushPattern()
1.911 +
1.912 +@pre The rendering target has been activated.
1.913 +@post Request to draw a rectangle with rounded corners has been accepted. There is no guarantee
1.914 + that the request has been processed when the method returns.
1.915 +
1.916 +@panic DGDI 7, if the rendering context has not been activated.
1.917 +@panic DGDI 9, if the brush style is EPatternedBrush but no pattern has been set.
1.918 +*/
1.919 +EXPORT_C void CDirectGdiContext::DrawRoundRect(const TRect& aRect, const TSize& aCornerSize)
1.920 + {
1.921 + GRAPHICS_TRACE("CDirectGdiContext::DrawRoundRect");
1.922 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.923 + GRAPHICS_ASSERT_ALWAYS(iBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.924 +
1.925 + TSize ellsize(aCornerSize);
1.926 + ellsize.iWidth <<= 1;
1.927 + ellsize.iHeight <<= 1;
1.928 +
1.929 + if (aRect.Width() > 0 && aRect.Height() > 0)
1.930 + {
1.931 + if (ellsize.iWidth < 3 || ellsize.iHeight < 3)
1.932 + {
1.933 + DrawRect(aRect);
1.934 + return;
1.935 + }
1.936 + if (aRect.Width() < ellsize.iWidth && aRect.Height() < ellsize.iHeight)
1.937 + {
1.938 + DrawEllipse(aRect);
1.939 + return;
1.940 + }
1.941 + iEngine->DrawRoundRect(aRect, aCornerSize);
1.942 + }
1.943 + }
1.944 +
1.945 +/**
1.946 +Draws a polyline using the points in an array. A polyline is a series of concatenated straight
1.947 +lines joining a set of points. Current pen settings and drawing mode applies. If @c aPointList
1.948 +has one element, a plot is performed.
1.949 +
1.950 +@param aPointList Array of points specifying points on the polyline.
1.951 +
1.952 +@pre The rendering target has been activated.
1.953 +@post Request to draw a polyline has been accepted.
1.954 + There is no guarantee that the request has been processed when the method returns.
1.955 + The internal drawing position is set to the last point in the array.
1.956 +
1.957 +@panic DGDI 7, if the rendering context has not been activated.
1.958 +
1.959 +@see CDirectGdiContext::DrawPolyLineNoEndPoint(const TArray<TPoint>&)
1.960 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.961 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.962 +@see CDirectGdiContext::SetPenColor(TRgb)
1.963 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.964 +*/
1.965 +EXPORT_C void CDirectGdiContext::DrawPolyLine(const TArray<TPoint>& aPointList)
1.966 + {
1.967 + GRAPHICS_TRACE("CDirectGdiContext::DrawPolyLine");
1.968 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.969 + if ((aPointList.Count() < 1) || iPenStyle == DirectGdi::ENullPen || (iPenSize.iWidth == 0) || (iPenSize.iHeight == 0))
1.970 + {
1.971 + return;
1.972 + }
1.973 +
1.974 + if (aPointList.Count() == 1)
1.975 + {
1.976 + Plot(aPointList[0]);
1.977 + }
1.978 + else
1.979 + {
1.980 + iEngine->DrawPolyLine(aPointList);
1.981 + }
1.982 + }
1.983 +
1.984 +
1.985 +/**
1.986 +Draws a polyline using the points in an array. A polyline is a series of concatenated straight
1.987 +lines joining a set of points. Current pen settings and drawing mode applies. If @c aPointList
1.988 +has less than two elements, no drawing is performed. If @c aPointList has exactly two elements
1.989 +then a DrawLine is performed.
1.990 +
1.991 +@param aPointList Array of points specifying points on the polyline.
1.992 +
1.993 +@pre The rendering target has been activated.
1.994 +@post Request to draw a polyline has been accepted.
1.995 + There is no guarantee that the request has been processed when the method returns.
1.996 + The internal drawing position is set to the last point in the array.
1.997 +
1.998 +@panic DGDI 7, if the rendering context has not been activated.
1.999 +
1.1000 +@see CDirectGdiContext::DrawPolyLine(const TArray<TPoint>&)
1.1001 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1002 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1003 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1004 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.1005 +*/
1.1006 +EXPORT_C void CDirectGdiContext::DrawPolyLineNoEndPoint(const TArray<TPoint>& aPointList)
1.1007 + {
1.1008 + GRAPHICS_TRACE("CDirectGdiContext::DrawPolyLineNoEndPoint");
1.1009 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1010 +
1.1011 + const TInt points = aPointList.Count();
1.1012 +
1.1013 + if (points == 1)
1.1014 + {
1.1015 + Plot(aPointList[0]);
1.1016 + }
1.1017 + else if (points == 2)
1.1018 + {
1.1019 + DrawLine(aPointList[0], aPointList[1]);
1.1020 + }
1.1021 + else if (points > 2 && !(iPenStyle == DirectGdi::ENullPen || (iPenSize.iWidth == 0) || (iPenSize.iHeight == 0)))
1.1022 + {
1.1023 + iEngine->DrawPolyLineNoEndPoint(aPointList);
1.1024 + }
1.1025 + }
1.1026 +
1.1027 +/**
1.1028 +Draws and fills a polygon defined using an array of points. The first point in the array defines the
1.1029 +start of the first side of the polygon. The final side of the polygon is drawn using the last point
1.1030 +from the array, and the line is drawn to the start point of the first side. The outline of the polygon
1.1031 +is drawn using the current pen settings and the area is filled with the current brush settings.
1.1032 +
1.1033 +Self-crossing polygons are filled according to the specified fill rule.
1.1034 +
1.1035 +If @c aPointList is empty, no drawing is performed. If it has one element, the result is the same as Plot().
1.1036 +If it has two elements, the result is the same as DrawLine().
1.1037 +
1.1038 +The error state is set to KErrArgument if aFillRule is an invalid fill rule.
1.1039 +
1.1040 +@param aPointList Array of points specifying the vertices of the polygon.
1.1041 +@param aFillRule Polygon filling rule.
1.1042 +
1.1043 +@pre The rendering target has been activated.
1.1044 +@post Request to draw a polygon has been accepted. There is no guarantee that the request has been processed
1.1045 + when the method returns. The internal drawing position is set to the last point in the array.
1.1046 +
1.1047 +@panic DGDI 7, if the rendering context has not been activated.
1.1048 +@panic DGDI 9, if the brush style is EPatternedBrush but no pattern has been set.
1.1049 +
1.1050 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1051 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1052 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1053 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1054 +@see CDirectGdiContext::SetBrushStyle(DirectGdi::TBrushStyle)
1.1055 +@see CDirectGdiContext::SetBrushPattern(const CFbsBitmap&)
1.1056 +@see CDirectGdiContext::SetBrushOrigin(TPoint)
1.1057 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.1058 +@see DirectGdi::TFillRule
1.1059 +*/
1.1060 +EXPORT_C void CDirectGdiContext::DrawPolygon(const TArray<TPoint>& aPointList, DirectGdi::TFillRule aFillRule)
1.1061 + {
1.1062 + GRAPHICS_TRACE("CDirectGdiContext::DrawPolygon");
1.1063 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1064 + GRAPHICS_ASSERT_ALWAYS(iBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.1065 +
1.1066 + if (aFillRule != DirectGdi::EAlternate && aFillRule != DirectGdi::EWinding)
1.1067 + {
1.1068 + iDriver.SetError(KErrArgument);
1.1069 + return;
1.1070 + }
1.1071 +
1.1072 + if (aPointList.Count() == 0)
1.1073 + {
1.1074 + return;
1.1075 + }
1.1076 +
1.1077 + iEngine->DrawPolygon(aPointList, aFillRule);
1.1078 + }
1.1079 +
1.1080 +
1.1081 +/**
1.1082 +Draws and fills an ellipse inside the given rectangle. Current pen and brush settings apply.
1.1083 +
1.1084 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1085 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1086 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1087 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1088 +@see CDirectGdiContext::SetBrushStyle(DirectGdi::TBrushStyle)
1.1089 +@see CDirectGdiContext::SetBrushPattern(const CFbsBitmap&)
1.1090 +
1.1091 +@param aRect The rectangle in which to draw the ellipse.
1.1092 +
1.1093 +@pre The rendering target has been activated.
1.1094 +@post Request to draw an ellipse has been accepted.
1.1095 + There is no guarantee that the request has been processed when the method returns.
1.1096 +
1.1097 +@panic DGDI 7, if the rendering context has not been activated.
1.1098 +@panic DGDI 9, if the brush style is EPatternedBrush but no pattern has been set.
1.1099 +*/
1.1100 +EXPORT_C void CDirectGdiContext::DrawEllipse(const TRect& aRect)
1.1101 + {
1.1102 + GRAPHICS_TRACE("CDirectGdiContext::DrawEllipse");
1.1103 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1104 + GRAPHICS_ASSERT_ALWAYS(iBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.1105 +
1.1106 + if (aRect.IsEmpty())
1.1107 + {
1.1108 + return;
1.1109 + }
1.1110 + iEngine->DrawEllipse(aRect);
1.1111 + }
1.1112 +
1.1113 +
1.1114 +/**
1.1115 +Draws a straight line from the start to the end position using current pen size, colour and style.
1.1116 +
1.1117 +@param aStart Start position.
1.1118 +@param aEnd End position.
1.1119 +
1.1120 +@pre The rendering target has been activated.
1.1121 +@post Request to draw a straight line with the current pen colour, size and style has been accepted.
1.1122 + There is no guarantee that the request has been processed when the method returns.
1.1123 + The internal drawing position is set to @c aEnd.
1.1124 +
1.1125 +@panic DGDI 7, if the rendering context has not been activated.
1.1126 +
1.1127 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1128 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1129 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1130 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.1131 +*/
1.1132 +EXPORT_C void CDirectGdiContext::DrawLine(const TPoint& aStart, const TPoint& aEnd)
1.1133 + {
1.1134 + GRAPHICS_TRACE("CDirectGdiContext::DrawLine");
1.1135 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1136 +
1.1137 + // Do not draw if start/end at the same point or pensize is 0
1.1138 + if(aStart == aEnd || iPenStyle == DirectGdi::ENullPen || iPenSize.iWidth == 0 || iPenSize.iHeight == 0)
1.1139 + {
1.1140 + return;
1.1141 + }
1.1142 +
1.1143 + iEngine->DrawLine(aStart, aEnd);
1.1144 + }
1.1145 +
1.1146 +
1.1147 +/**
1.1148 +Draws a straight line from the current internal drawing position to a point using the current pen size, colour and style.
1.1149 +
1.1150 +@param aPoint The end-point of the line.
1.1151 +
1.1152 +@pre Rendering target has been activated.
1.1153 +@post Request to draw a straight line to a specified point with the current pen colour, size and style has been accepted.
1.1154 + There is no guarantee that the request has been processed when the method returns.
1.1155 + Internal drawing position is set to @c aPoint.
1.1156 +
1.1157 +@panic DGDI 7, if the rendering context has not been activated.
1.1158 +
1.1159 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1160 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1161 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1162 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.1163 +@see CDirectGdiContext::MoveTo(TPoint)
1.1164 +@see CDirectGdiContext::MoveBy(TPoint)
1.1165 +*/
1.1166 +EXPORT_C void CDirectGdiContext::DrawLineTo(const TPoint& aPoint)
1.1167 + {
1.1168 + GRAPHICS_TRACE("CDirectGdiContext::DrawLineTo");
1.1169 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1170 + if(iPenStyle == DirectGdi::ENullPen || iPenSize.iWidth == 0 || iPenSize.iHeight == 0)
1.1171 + {
1.1172 + return;
1.1173 + }
1.1174 + iEngine->DrawLineTo (aPoint);
1.1175 + }
1.1176 +
1.1177 +
1.1178 +/**
1.1179 +Draws a straight line relative to the current internal drawing position, using a vector.
1.1180 +The start point of the line is the current internal drawing position. The vector @c aVector
1.1181 +is added to the internal drawing position to give the end point of the line.
1.1182 +
1.1183 +@param aVector The vector to add to the current internal drawing position, giving the end point of the line.
1.1184 +
1.1185 +@pre The rendering target has been activated.
1.1186 +@post The request to draw a straight line using the vector with current pen colour, size and style has been
1.1187 + accepted. There is no guarantee that the request has been processed when the method returns.
1.1188 + The internal drawing position is set to the end of the line.
1.1189 +
1.1190 +@panic DGDI 7, if the rendering context has not been activated.
1.1191 +
1.1192 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1193 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1194 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1195 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.1196 +@see CDirectGdiContext::MoveTo(TPoint)
1.1197 +@see CDirectGdiContext::MoveBy(TPoint)
1.1198 +*/
1.1199 +EXPORT_C void CDirectGdiContext::DrawLineBy(const TPoint& aVector)
1.1200 + {
1.1201 + GRAPHICS_TRACE("CDirectGdiContext::DrawLineBy");
1.1202 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1203 + if(iPenStyle == DirectGdi::ENullPen || iPenSize.iWidth == 0 || iPenSize.iHeight == 0)
1.1204 + {
1.1205 + return;
1.1206 + }
1.1207 +
1.1208 + if (aVector != TPoint(0,0))
1.1209 + {
1.1210 + iEngine->DrawLineBy(aVector);
1.1211 + }
1.1212 + }
1.1213 +
1.1214 +
1.1215 +/**
1.1216 +Draws and fills a rectangle. The outlines are drawn according to the current pen colour, size and style.
1.1217 +The area inside the rectangle is filled according to the current brush colour and style.
1.1218 +
1.1219 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.1220 +@see CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle)
1.1221 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1222 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1223 +@see CDirectGdiContext::SetBrushStyle(DirectGdi::TBrushStyle)
1.1224 +@see CDirectGdiContext::SetBrushPattern(const CFbsBitmap&)
1.1225 +
1.1226 +@param aRect The rectangle.
1.1227 +
1.1228 +@pre The rendering target has been activated.
1.1229 +@post Request to draw a rectangle according to the current pen and brush settings has been accepted.
1.1230 + There is no guarantee that the request has been processed when the method returns.
1.1231 +
1.1232 +@panic DGDI 7, if the rendering context has not been activated.
1.1233 +@panic DGDI 9, if the brush style is EPatternedBrush but no pattern has been set.
1.1234 +*/
1.1235 +EXPORT_C void CDirectGdiContext::DrawRect(const TRect& aRect)
1.1236 + {
1.1237 + GRAPHICS_TRACE("CDirectGdiContext::DrawRect");
1.1238 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1239 + GRAPHICS_ASSERT_ALWAYS(iBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.1240 +
1.1241 + if(aRect.IsEmpty())
1.1242 + return;
1.1243 +
1.1244 + iEngine->DrawRect(aRect);
1.1245 + }
1.1246 +
1.1247 +
1.1248 +/**
1.1249 +Draws text at the last print position.
1.1250 +
1.1251 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1252 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1253 +@see CDirectGdiContext::SetFont(const CFont*)
1.1254 +
1.1255 +@param aText The text string to be drawn.
1.1256 +@param aParam Parameters used in drawing text.
1.1257 +
1.1258 +@pre The rendering target has been activated.
1.1259 +@post Request to draw the text at the last text position using the current font and pen colour has been accepted.
1.1260 + There is no guarantee that the request has been processed when the method returns.
1.1261 +
1.1262 +@panic DGDI 7, if the rendering context has not been activated.
1.1263 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush
1.1264 + style is neither ENullBrush nor ESolidBrush.
1.1265 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1266 +*/
1.1267 +EXPORT_C void CDirectGdiContext::DrawText(const TDesC& aText, const DirectGdi::TTextParameters* aParam)
1.1268 + {
1.1269 + GRAPHICS_TRACE("CDirectGdiContext::DrawText");
1.1270 + DrawText(aText, aParam, iLastPrintPosition, DirectGdi::ELeft, CFont::EHorizontal);
1.1271 + }
1.1272 +
1.1273 +
1.1274 +/**
1.1275 +Draws text at the specified text position.
1.1276 +
1.1277 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1278 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1279 +@see CDirectGdiContext::SetFont(const CFont*)
1.1280 +
1.1281 +@param aText The text string to be drawn.
1.1282 +@param aParam Parameters used in drawing text.
1.1283 +@param aPosition The position to draw at.
1.1284 +
1.1285 +@pre The rendering target has been activated.
1.1286 +@post Request to draw the text at the specified position using the current font and pen colour has been accepted.
1.1287 + There is no guarantee that the request has been processed when the method returns.
1.1288 +@panic DGDI 7, if the rendering context has not been activated.
1.1289 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush
1.1290 + style is neither ENullBrush nor ESolidBrush.
1.1291 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1292 +*/
1.1293 +EXPORT_C void CDirectGdiContext::DrawText(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TPoint& aPosition)
1.1294 + {
1.1295 + GRAPHICS_TRACE("CDirectGdiContext::DrawText");
1.1296 + DrawText(aText, aParam, aPosition, DirectGdi::ELeft, CFont::EHorizontal);
1.1297 + }
1.1298 +
1.1299 +
1.1300 +/**
1.1301 +Draws text clipped to the specified rectangle.
1.1302 +
1.1303 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1304 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1305 +@see CDirectGdiContext::SetFont(const CFont*)
1.1306 +
1.1307 +@param aText The text string to be drawn.
1.1308 +@param aParam Parameters used in drawing text.
1.1309 +@param aClipRect The clipping rectangle.
1.1310 +
1.1311 +@pre The rendering target has been activated.
1.1312 +@post Request to draw the text at the last text position using the current font and pen colour, clipped to the specified rectangle has been accepted.
1.1313 + There is no guarantee that the request has been processed when the method returns.
1.1314 +@panic DGDI 7, if the rendering context has not been activated.
1.1315 +@panic DGDI 10, if the active font is an outline and/or shadow font and
1.1316 + the brush style is neither ENullBrush nor ESolidBrush.
1.1317 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1318 +*/
1.1319 +EXPORT_C void CDirectGdiContext::DrawText(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect& aClipRect)
1.1320 + {
1.1321 + GRAPHICS_TRACE("CDirectGdiContext::DrawText");
1.1322 + DrawText(aText, aParam, iLastPrintPosition, DirectGdi::ELeft, CFont::EHorizontal, &aClipRect);
1.1323 + }
1.1324 +
1.1325 +
1.1326 +/**
1.1327 +Draws text clipped to the specified filled rectangle using a baseline offset,
1.1328 +horizontal alignment and a margin.
1.1329 +
1.1330 +@see CDirectGdiContext::SetPenColor(TRgb)
1.1331 +@see CDirectGdiContext::SetBrushColor(TRgb)
1.1332 +@see CDirectGdiContext::SetFont(const CFont*)
1.1333 +
1.1334 +@param aText The text string to be drawn.
1.1335 +@param aParam Parameters used in drawing text.
1.1336 +@param aClipFillRect The clipping rectangle (this rect will also be filled before text is plotted).
1.1337 +@param aBaselineOffset An offset in pixels for the baseline from the normal position (bottom of the rectangle minus the descent of the font).
1.1338 +@param aAlignment Horizontal alignment option relative to the specified rectangle.
1.1339 +@param aMargin Offset to add to the position as calculated using specified rectangle.
1.1340 +
1.1341 +@pre The rendering target has been activated.
1.1342 +@post Request to draw the text within the filled rectangle using the current font and pen colour, offset and clipped to the specified rectangle has been accepted.
1.1343 + There is no guarantee that the request has been processed when the method returns.
1.1344 +@panic DGDI 7, if the rendering context has not been activated.
1.1345 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush
1.1346 + style is neither ENullBrush nor ESolidBrush.
1.1347 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1348 +*/
1.1349 +EXPORT_C void CDirectGdiContext::DrawText(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect& aClipFillRect, TInt aBaselineOffset, DirectGdi::TTextAlign aAlignment, TInt aMargin)
1.1350 + {
1.1351 + GRAPHICS_TRACE("CDirectGdiContext::DrawText");
1.1352 + TPoint p(aClipFillRect.iTl);
1.1353 + p.iY += aBaselineOffset;
1.1354 + switch (aAlignment)
1.1355 + {
1.1356 + case DirectGdi::ELeft:
1.1357 + {
1.1358 + p.iX += aMargin;
1.1359 + break;
1.1360 + }
1.1361 + case DirectGdi::ERight:
1.1362 + {
1.1363 + p.iX = aClipFillRect.iBr.iX - aMargin;
1.1364 + break;
1.1365 + }
1.1366 + case DirectGdi::ECenter:
1.1367 + {
1.1368 + p.iX += (aClipFillRect.Width() >> 1) + aMargin;
1.1369 + break;
1.1370 + }
1.1371 + default:
1.1372 + {
1.1373 + iDriver.SetError(KErrArgument);
1.1374 + return;
1.1375 + }
1.1376 + }
1.1377 + DrawText(aText, aParam, p, aAlignment, CFont::EHorizontal, &aClipFillRect, &aClipFillRect);
1.1378 + }
1.1379 +
1.1380 +
1.1381 +/**
1.1382 +The private general DrawText routine that implements all the others.
1.1383 +
1.1384 +@param aText The text to be drawn.
1.1385 +@param aParam Parameters used in drawing text.
1.1386 +@param aPosition The origin of the text.
1.1387 +@param aAlignment Left, centred or right, around aPosition; not used if drawing vertically.
1.1388 +@param aDirection Direction: left to right, right to left, or top to bottom.
1.1389 +@param aClipRect If non-null, used as a clippingrect when the text is drawn.
1.1390 +@param aFillRect If non-null, filled before the text is drawn.
1.1391 +
1.1392 +@panic DGDI 7, if the rendering context has not been activated.
1.1393 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush
1.1394 + style is neither ENullBrush nor ESolidBrush.
1.1395 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1396 +*/
1.1397 +void CDirectGdiContext::DrawText(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TPoint& aPosition, DirectGdi::TTextAlign aAlignment,
1.1398 + CFont::TTextDirection aDirection, const TRect* aClipRect, const TRect* aFillRect)
1.1399 + {
1.1400 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1401 + // anything to do?
1.1402 + if (aClipRect && aClipRect->IsEmpty())
1.1403 + {
1.1404 + iDriver.SetError(KErrArgument);
1.1405 + return;
1.1406 + }
1.1407 +
1.1408 + GRAPHICS_ASSERT_ALWAYS(iFont.Handle() != 0, EDirectGdiPanicNoFontSelected);
1.1409 + // This next check actually covers both bitmap and open fonts
1.1410 + const CBitmapFont* bitmap_font = iFont.Address();
1.1411 + GRAPHICS_ASSERT_ALWAYS(bitmap_font != 0, EDirectGdiPanicNoFontSelected);
1.1412 +
1.1413 + // measure the text
1.1414 + CFont::TMeasureTextInput measure_text_input;
1.1415 + measure_text_input.iCharJustNum = iCharJustNum;
1.1416 + measure_text_input.iCharJustExcess = iCharJustExcess;
1.1417 + measure_text_input.iWordJustNum = iWordJustNum;
1.1418 + measure_text_input.iWordJustExcess = iWordJustExcess;
1.1419 + measure_text_input.iFlags |= CFont::TMeasureTextInput::EFVisualOrder;
1.1420 + if (aParam)
1.1421 + {
1.1422 + GRAPHICS_ASSERT_ALWAYS(aParam->iStart < aParam->iEnd ,EDirectGdiPanicBadParameter);
1.1423 + measure_text_input.iStartInputChar = aParam->iStart;
1.1424 + measure_text_input.iEndInputChar = Min(aText.Length(),aParam->iEnd);
1.1425 + }
1.1426 + CFont::TMeasureTextOutput measure_text_output;
1.1427 + const TInt advance = iFont.MeasureText(aText, &measure_text_input, &measure_text_output);
1.1428 + TRect text_bounds = measure_text_output.iBounds;
1.1429 +
1.1430 + //for linked fonts need an adjustment to the underline postion
1.1431 + TInt underlineStrikeoutOffset = BaselineCorrection();
1.1432 +
1.1433 + if (iUnderline == DirectGdi::EUnderlineOn)
1.1434 + {
1.1435 + TInt underline_top = 0;
1.1436 + TInt underline_bottom = 0;
1.1437 + GetUnderlineMetrics(underline_top, underline_bottom);
1.1438 + underline_top+=underlineStrikeoutOffset;
1.1439 + underline_bottom+=underlineStrikeoutOffset;
1.1440 + text_bounds.iTl.iY = Min(text_bounds.iTl.iY, underline_top);
1.1441 + text_bounds.iBr.iY = Max(text_bounds.iBr.iY, underline_bottom);
1.1442 + }
1.1443 + if (iStrikethrough == DirectGdi::EStrikethroughOn)
1.1444 + {
1.1445 + TInt strike_top = 0;
1.1446 + TInt strike_bottom = 0;
1.1447 + GetStrikethroughMetrics(strike_top, strike_bottom);
1.1448 + strike_top+=underlineStrikeoutOffset;
1.1449 + strike_bottom+=underlineStrikeoutOffset;
1.1450 + text_bounds.iTl.iY = Min(text_bounds.iTl.iY, strike_top);
1.1451 + text_bounds.iBr.iY = Max(text_bounds.iBr.iY, strike_bottom);
1.1452 + }
1.1453 + if (iUnderline == DirectGdi::EUnderlineOn || iStrikethrough == DirectGdi::EStrikethroughOn)
1.1454 + {
1.1455 + if (aDirection == CFont::EHorizontal)
1.1456 + {
1.1457 + text_bounds.iTl.iX = Min(text_bounds.iTl.iX, 0);
1.1458 + text_bounds.iBr.iX = Max(text_bounds.iBr.iX, advance);
1.1459 + }
1.1460 + else
1.1461 + {
1.1462 + text_bounds.iTl.iY = Min(text_bounds.iTl.iY, 0);
1.1463 + text_bounds.iBr.iY = Max(text_bounds.iBr.iY, advance);
1.1464 + }
1.1465 + }
1.1466 +
1.1467 + // work out the text origin and new drawing position
1.1468 + TPoint text_origin = aPosition;
1.1469 + if (aDirection != CFont::EVertical)
1.1470 + {
1.1471 + const TInt leftSideBearing = Min(text_bounds.iTl.iX, 0);
1.1472 + const TInt rightSideBearing = Max(text_bounds.iBr.iX, advance);
1.1473 + switch (aAlignment)
1.1474 + {
1.1475 + // We are forbidding side-bearings to leak over the sides here,
1.1476 + // but still keeping the start and end pen positions within bounds.
1.1477 + case DirectGdi::ELeft:
1.1478 + text_origin.iX -= leftSideBearing;
1.1479 + break;
1.1480 + case DirectGdi::ERight:
1.1481 + text_origin.iX -= rightSideBearing;
1.1482 + break;
1.1483 + case DirectGdi::ECenter:
1.1484 + // Centre is the average of left and right
1.1485 + text_origin.iX -= (leftSideBearing + rightSideBearing) >> 1;
1.1486 + break;
1.1487 + default:
1.1488 + iDriver.SetError(KErrArgument);
1.1489 + return;
1.1490 + }
1.1491 + }
1.1492 + iLastPrintPosition = text_origin;
1.1493 + if (aDirection == CFont::EHorizontal)
1.1494 + {
1.1495 + iLastPrintPosition.iX += advance;
1.1496 + }
1.1497 + else
1.1498 + {
1.1499 + iLastPrintPosition.iY += advance;
1.1500 + }
1.1501 + text_origin.iY += bitmap_font->iAlgStyle.iBaselineOffsetInPixels;
1.1502 + text_bounds.Move(text_origin);
1.1503 + text_origin += iOrigin;
1.1504 +
1.1505 + // determine clipping rectangle
1.1506 + TRect clipRect = aClipRect ? *aClipRect : text_bounds;
1.1507 +
1.1508 + // fill the box if necessary
1.1509 + if (aFillRect && (iBrushStyle != DirectGdi::ENullBrush))
1.1510 + {
1.1511 + TRect fillBox = *aFillRect;
1.1512 + if (fillBox.Intersects(clipRect))
1.1513 + {
1.1514 + fillBox.Intersection(clipRect);
1.1515 + iEngine->SetPenStyle(DirectGdi::ENullPen); // Fill box, don't outline it
1.1516 + iEngine->DrawRect(fillBox);
1.1517 + iEngine->SetPenStyle(iPenStyle); // Put the pen style back
1.1518 + }
1.1519 + }
1.1520 + if (!aText.Length())
1.1521 + {
1.1522 + return;
1.1523 + }
1.1524 +
1.1525 + clipRect.Move(iOrigin);
1.1526 +
1.1527 + // decide which drawing routine to call
1.1528 +
1.1529 + TOpenFontMetrics metrics;
1.1530 + iFont.GetFontMetrics(metrics);
1.1531 + const TInt maxwidth = metrics.MaxWidth();
1.1532 + // extext will be TRUE, if font is underline/strikethrough/anti-aliased or it has shadow/outline effects ON.
1.1533 + // Depending on these properties it will call the proper draw routine.
1.1534 + TBool extext = EFalse;
1.1535 + TBool normaltext = EFalse;
1.1536 + const TBool antiAliased = (bitmap_font->GlyphBitmapType() == EAntiAliasedGlyphBitmap);
1.1537 + const TBool outlineAndShadow = (bitmap_font->GlyphBitmapType() == EFourColourBlendGlyphBitmap);
1.1538 + if (antiAliased || outlineAndShadow )
1.1539 + {
1.1540 + if ((outlineAndShadow) && !((iBrushStyle == DirectGdi::ENullBrush) || (iBrushStyle == DirectGdi::ESolidBrush)))
1.1541 + {
1.1542 + //For future compatibility it is better if brush style of ENullBrush or ESolidBrush is used
1.1543 + //when drawing outline and shadow fonts.
1.1544 + GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidBrushStyle);
1.1545 + }
1.1546 + extext = ETrue;
1.1547 + }
1.1548 + else if ((iUnderline == DirectGdi::EUnderlineOn) || (iStrikethrough == DirectGdi::EStrikethroughOn) || (iCharJustNum > 0) || (iWordJustNum > 0))
1.1549 + extext = ETrue;
1.1550 + else
1.1551 + normaltext = ETrue;
1.1552 +
1.1553 + const TInt charjustexcess = iCharJustExcess;
1.1554 + const TInt charjustnum = iCharJustNum;
1.1555 + const TInt wordjustexcess = iWordJustExcess;
1.1556 + const TInt wordjustnum = iWordJustNum;
1.1557 +
1.1558 + // Set up the parameter block for character positioning.
1.1559 + CFont::TPositionParam param;
1.1560 + param.iDirection = static_cast<TInt16>(aDirection);
1.1561 + param.iText.Set(aText);
1.1562 + TInt endDraw = aText.Length();
1.1563 + if (aParam)
1.1564 + {
1.1565 + param.iPosInText = aParam->iStart;
1.1566 + endDraw = Min(aText.Length(),aParam->iEnd);
1.1567 + }
1.1568 + else
1.1569 + {
1.1570 + param.iPosInText = 0;
1.1571 + }
1.1572 + param.iPen = text_origin;
1.1573 +
1.1574 + // Draw the text.
1.1575 + if (normaltext)
1.1576 + {
1.1577 + DoDrawText(param, endDraw, clipRect);
1.1578 + }
1.1579 + else if (extext)
1.1580 + {
1.1581 + DoDrawTextEx(param, endDraw, clipRect,underlineStrikeoutOffset);
1.1582 + }
1.1583 +
1.1584 + // Reset the justification parameters to their original values.
1.1585 + // These will be updated as required later in code.
1.1586 + iCharJustExcess = charjustexcess;
1.1587 + iCharJustNum = charjustnum;
1.1588 + iWordJustExcess = wordjustexcess;
1.1589 + iWordJustNum = wordjustnum;
1.1590 +
1.1591 + if (iAutoUpdateJustification)
1.1592 + UpdateJustification(aText, aParam);
1.1593 + }
1.1594 +
1.1595 +/**
1.1596 +Overridden function which draws monochrome text within the given clip rectangle. No rotation applied.
1.1597 +@param aParam Defines glyph code, ligature creation and diacritic placement.
1.1598 +@param aEnd The end position within the text descriptor to draw.
1.1599 +@param aClipRect If not-empty, used as a clippingrect when the text is drawn.
1.1600 +*/
1.1601 +void CDirectGdiContext::DoDrawText(CFont::TPositionParam& aParam, const TInt aEnd, const TRect& aClipRect)
1.1602 + {
1.1603 + iEngine->BeginDrawGlyph();
1.1604 + RShapeInfo shapeInfo;
1.1605 + while (aParam.iPosInText < aEnd)
1.1606 + {
1.1607 + if (iFont.GetCharacterPosition2(aParam, shapeInfo))
1.1608 + {
1.1609 + const CFont::TPositionParam::TOutput* output = aParam.iOutput;
1.1610 + for (TInt i = 0; i < aParam.iOutputGlyphs; ++i, ++output)
1.1611 + iEngine->DrawGlyph(output->iBounds.iTl, output->iCode, output->iBitmap, EMonochromeGlyphBitmap, output->iBitmapSize, aClipRect); // All other parameters are default
1.1612 + }
1.1613 + }
1.1614 + iEngine->EndDrawGlyph();
1.1615 +
1.1616 + if (shapeInfo.IsOpen())
1.1617 + shapeInfo.Close();
1.1618 + }
1.1619 +
1.1620 +
1.1621 +/**
1.1622 +Overridden function which draws monochrome text within the given clip rectangle.
1.1623 +The current rotation and font style (strikethrough, underline) are applied.
1.1624 +
1.1625 +@param aParam Defines glyph code, ligature creation and diacritic placement.
1.1626 +@param aEnd The end position within the text descriptor to draw.
1.1627 +@param aClipRect If not-empty, used as a clipping rect when the text is drawn.
1.1628 +@param aUnderlineStrikethroughOffset the offset for the underline, passed to save calculating this value again
1.1629 +*/
1.1630 +void CDirectGdiContext::DoDrawTextEx(CFont::TPositionParam& aParam, const TInt aEnd, const TRect& aClipRect, const TInt aUnderlineStrikethroughOffset)
1.1631 + {
1.1632 + TPoint startPen = aParam.iPen;
1.1633 + const CBitmapFont* bitmap_font = iFont.Address();
1.1634 + TInt underlineTop = 0;
1.1635 + TInt underlineBottom = 0;
1.1636 + if (iUnderline == DirectGdi::EUnderlineOn)
1.1637 + {
1.1638 + GetUnderlineMetrics(underlineTop, underlineBottom);
1.1639 + underlineTop+=aUnderlineStrikethroughOffset;
1.1640 + underlineBottom+=aUnderlineStrikethroughOffset;
1.1641 + }
1.1642 + TInt strikeTop = 0;
1.1643 + TInt strikeBottom = 0;
1.1644 + if (iStrikethrough == DirectGdi::EStrikethroughOn)
1.1645 + {
1.1646 + GetStrikethroughMetrics(strikeTop, strikeBottom);
1.1647 + strikeTop+=aUnderlineStrikethroughOffset;
1.1648 + strikeBottom+=aUnderlineStrikethroughOffset;
1.1649 + }
1.1650 +
1.1651 + iEngine->BeginDrawGlyph();
1.1652 + RShapeInfo shapeInfo;
1.1653 + while (aParam.iPosInText < aEnd)
1.1654 + {
1.1655 + if (!iFont.GetCharacterPosition2(aParam, shapeInfo))
1.1656 + {
1.1657 + continue;
1.1658 + }
1.1659 +
1.1660 + TInt adjustment = 0;
1.1661 + if ((iCharJustExcess > 0) && (iCharJustNum > 0)) // character clipping/justification
1.1662 + {
1.1663 + adjustment = CGraphicsContext::JustificationInPixels(iCharJustExcess, iCharJustNum);
1.1664 + }
1.1665 +
1.1666 + const CFont::TPositionParam::TOutput* output = aParam.iOutput;
1.1667 + for (TInt i = 0; i < aParam.iOutputGlyphs; ++i, ++output)
1.1668 + {
1.1669 + //get the character metrics for the glyph type
1.1670 + TOpenFontCharMetrics characterParams;
1.1671 + const TUint8* bitmap;
1.1672 + TSize size;
1.1673 + //note may now be using a glyph code, and not a character
1.1674 + iFont.GetCharacterData(aParam.iOutput[i].iCode,characterParams,bitmap,size);
1.1675 + TGlyphBitmapType glyphType = characterParams.GlyphType();
1.1676 +
1.1677 + switch (glyphType)
1.1678 + {
1.1679 + case EAntiAliasedGlyphBitmap:
1.1680 + case EFourColourBlendGlyphBitmap:
1.1681 + case EDefaultGlyphBitmap:
1.1682 + iEngine->DrawGlyph(output->iBounds.iTl, output->iCode, output->iBitmap, glyphType, output->iBitmapSize, aClipRect);
1.1683 + break;
1.1684 +
1.1685 + default:
1.1686 + //if the outline or shadow is not specified for the character, then use the font setting for the glyph bitmap type
1.1687 + iEngine->DrawGlyph(output->iBounds.iTl, output->iCode, output->iBitmap,
1.1688 + bitmap_font->GlyphBitmapType(), output->iBitmapSize, aClipRect);
1.1689 + break;
1.1690 + }
1.1691 + }
1.1692 +
1.1693 + if (adjustment)
1.1694 + {
1.1695 + aParam.iPen.iX += adjustment;
1.1696 + }
1.1697 + if ((iWordJustExcess > 0) && (iWordJustNum > 0) && (aParam.iOutput[0].iCode == 0x0020)) // word justification
1.1698 + {
1.1699 + adjustment = CGraphicsContext::JustificationInPixels(iWordJustExcess, iWordJustNum);
1.1700 + aParam.iPen.iX += adjustment;
1.1701 + }
1.1702 + }
1.1703 + iEngine->EndDrawGlyph();
1.1704 + if (shapeInfo.IsOpen())
1.1705 + shapeInfo.Close();
1.1706 +
1.1707 + if (iUnderline == DirectGdi::EUnderlineOn)
1.1708 + {
1.1709 + TRect underlineRect(startPen.iX, startPen.iY + underlineTop, aParam.iPen.iX, startPen.iY + underlineBottom);
1.1710 + FillRect(underlineRect, iPenColor, aClipRect);
1.1711 + }
1.1712 +
1.1713 + if (iStrikethrough == DirectGdi::EStrikethroughOn)
1.1714 + {
1.1715 + TRect strikethroughRect(startPen.iX, startPen.iY + strikeTop, aParam.iPen.iX, startPen.iY + strikeBottom);
1.1716 + FillRect(strikethroughRect, iPenColor, aClipRect);
1.1717 + }
1.1718 + }
1.1719 +
1.1720 +
1.1721 +/**
1.1722 +Fills the given rectangle with the specified colour (subject to the clip rect).
1.1723 +This function is internal and used by the text drawing routines.
1.1724 +
1.1725 +@param aRect The rectangle to fill.
1.1726 +@param aColor The colour to fill it with.
1.1727 +@param aClipRect The clipping rect.
1.1728 +*/
1.1729 +void CDirectGdiContext::FillRect(const TRect& aRect, const TRgb& aColor, const TRect& aClipRect)
1.1730 + {
1.1731 + TRect fillRect = aRect;
1.1732 + if (fillRect.Intersects(aClipRect))
1.1733 + {
1.1734 + fillRect.Intersection(aClipRect);
1.1735 + // Override the current settings temporarily
1.1736 + iEngine->SetBrushColor(aColor);
1.1737 + iEngine->SetBrushStyle(DirectGdi::ESolidBrush);
1.1738 + iEngine->SetPenStyle(DirectGdi::ENullPen); // Fill box, don't outline it
1.1739 + fillRect.Move(-iOrigin);
1.1740 + iEngine->DrawRect(fillRect);
1.1741 + // Put things back
1.1742 + iEngine->SetPenStyle(iPenStyle);
1.1743 + iEngine->SetBrushStyle(iBrushStyle);
1.1744 + iEngine->SetBrushColor(iBrushColor);
1.1745 + }
1.1746 + }
1.1747 +
1.1748 +
1.1749 +/**
1.1750 +Draws text at the last print position and then rotates it into a vertical position.
1.1751 +
1.1752 +@param aText The text string to be drawn.
1.1753 +@param aParam Parameters used in drawing text.
1.1754 +@param aUp If ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1755 +
1.1756 +@panic DGDI 7, if the rendering context has not been activated.
1.1757 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush
1.1758 + style is neither ENullBrush nor ESolidBrush.
1.1759 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1760 +*/
1.1761 +EXPORT_C void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, TBool aUp)
1.1762 + {
1.1763 + GRAPHICS_TRACE("CDirectGdiContext::DrawTextVertical");
1.1764 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1765 + // This next check covers both bitmap and open fonts
1.1766 + const CBitmapFont* bitmapFont = iFont.Address();
1.1767 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.1768 + TRect clipRect2(0, 0, 0, 0);
1.1769 + TInt baselineOffset = 0;
1.1770 + TInt margin = 0;
1.1771 + CalculateClipRect2PlusBaselineOffsetAndMargin(aText, aParam, iLastPrintPosition, aUp, clipRect2, baselineOffset, margin);
1.1772 + DrawTextVertical(aText, aParam, NULL, &clipRect2, NULL, baselineOffset, -1, aUp, DirectGdi::ELeft, margin); //-1 signifies that text will not be clipped
1.1773 + }
1.1774 +
1.1775 +
1.1776 +/**
1.1777 +Draws text vertically from the specified position.
1.1778 +
1.1779 +@param aText The text string to be drawn.
1.1780 +@param aParam Parameters used in drawing text.
1.1781 +@param aPosition A point specifying the position of the left end of the text.
1.1782 +@param aUp If ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1783 +
1.1784 +@panic DGDI 7, if the rendering context has not been activated.
1.1785 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush style is neither
1.1786 + ENullBrush nor ESolidBrush.
1.1787 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1788 +*/
1.1789 +EXPORT_C void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TPoint& aPosition, TBool aUp)
1.1790 + {
1.1791 + GRAPHICS_TRACE("CDirectGdiContext::DrawTextVertical");
1.1792 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1793 + // This next check covers both bitmap and open fonts
1.1794 + const CBitmapFont* bitmapFont = iFont.Address();
1.1795 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.1796 + TRect clipRect2(0, 0, 0, 0);
1.1797 + TInt baselineOffset = 0;
1.1798 + TInt margin = 0;
1.1799 + CalculateClipRect2PlusBaselineOffsetAndMargin(aText, aParam, aPosition, aUp, clipRect2, baselineOffset, margin);
1.1800 + DrawTextVertical(aText, aParam, NULL, &clipRect2, NULL, baselineOffset, -1, aUp, DirectGdi::ELeft, margin);//-1 signifies that text will not be clipped
1.1801 + }
1.1802 +
1.1803 +
1.1804 +/**
1.1805 +Draws text clipped to the specified rectangle and then rotates it into a vertical position.
1.1806 +
1.1807 +@param aText The text string to be drawn.
1.1808 +@param aParam Parameters used in drawing text.
1.1809 +@param aClipRect The clipping rectangle.
1.1810 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1811 +
1.1812 +@panic DGDI 7, if the rendering context has not been activated.
1.1813 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush style is neither ENullBrush
1.1814 + nor ESolidBrush.
1.1815 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1816 +*/
1.1817 +EXPORT_C void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect& aClipRect, TBool aUp)
1.1818 + {
1.1819 + GRAPHICS_TRACE("CDirectGdiContext::DrawTextVertical");
1.1820 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1821 + // This next check covers both bitmap and open fonts
1.1822 + const CBitmapFont* bitmapFont = iFont.Address();
1.1823 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.1824 + TRect clipRect2(0, 0, 0, 0);
1.1825 + TInt baselineOffset = 0;
1.1826 + TInt margin = 0;
1.1827 + CalculateClipRect2PlusBaselineOffsetAndMargin(aText, aParam, iLastPrintPosition, aUp, clipRect2, baselineOffset, margin);
1.1828 + DrawTextVertical(aText, aParam, &aClipRect, &clipRect2, NULL, baselineOffset, -1, aUp, DirectGdi::ELeft, margin);
1.1829 + }
1.1830 +
1.1831 +
1.1832 +/**
1.1833 +Private internal function for calculating several parameters needed by these routines.
1.1834 +
1.1835 +@param aText The text string to be drawn.
1.1836 +@param aParam Parameters used in drawing text.
1.1837 +@param aPosition A point specifying the position of the left end of the text.
1.1838 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1839 +@param aClipRect2 On return, contains clipping rectangle.
1.1840 +@param aBaselineOffset On return, contains baseline offset.
1.1841 +@param aMargin On return, contains margin.
1.1842 +*/
1.1843 +void CDirectGdiContext::CalculateClipRect2PlusBaselineOffsetAndMargin(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TPoint& aPosition, TBool aUp, TRect& aClipRect2, TInt& aBaselineOffset, TInt& aMargin)
1.1844 + {
1.1845 + TOpenFontMetrics metrics;
1.1846 + iFont.GetFontMetrics(metrics);
1.1847 + aBaselineOffset = metrics.MaxHeight();
1.1848 + TInt height = aBaselineOffset + metrics.MaxDepth();
1.1849 + // The next few lines do much the same as TextWidthInPixels but pass
1.1850 + // the text in visual order instead of logical order and also take
1.1851 + // full account of left and right side bearings on the text
1.1852 + CFont::TMeasureTextOutput output;
1.1853 + CFont::TMeasureTextInput input;
1.1854 + input.iFlags = CFont::TMeasureTextInput::EFVisualOrder;
1.1855 + if (aParam)
1.1856 + {
1.1857 + GRAPHICS_ASSERT_ALWAYS(aParam->iStart < aParam->iEnd ,EDirectGdiPanicBadParameter);
1.1858 + input.iStartInputChar = aParam->iStart;
1.1859 + input.iEndInputChar = Min(aText.Length(),aParam->iEnd);
1.1860 + }
1.1861 + TInt advance = iFont.MeasureText(aText, &input, &output);
1.1862 + TInt leftBearing = output.iBounds.iTl.iX;
1.1863 + TInt rightBearing = advance - output.iBounds.iBr.iX;
1.1864 + aMargin = 0;
1.1865 + if (aUp)
1.1866 + {
1.1867 + aClipRect2.iTl.iX = aPosition.iX - aBaselineOffset;
1.1868 + aClipRect2.iTl.iY = aPosition.iY - advance;
1.1869 + aClipRect2.iBr.iX = aPosition.iX + height - aBaselineOffset + 1;
1.1870 + aClipRect2.iBr.iY = aPosition.iY;
1.1871 + if (leftBearing < 0)
1.1872 + {
1.1873 + aClipRect2.iBr.iY -= leftBearing;
1.1874 + aMargin = -leftBearing;
1.1875 + }
1.1876 + if (rightBearing < 0)
1.1877 + {
1.1878 + aClipRect2.iTl.iY += rightBearing;
1.1879 + }
1.1880 + }
1.1881 + else
1.1882 + {
1.1883 + aClipRect2.iTl.iX = aPosition.iX + aBaselineOffset- height;
1.1884 + aClipRect2.iTl.iY = aPosition.iY;
1.1885 + aClipRect2.iBr.iX = aPosition.iX + aBaselineOffset + 1;
1.1886 + aClipRect2.iBr.iY = aPosition.iY + advance;
1.1887 + if (leftBearing < 0)
1.1888 + {
1.1889 + aClipRect2.iTl.iY += leftBearing;
1.1890 + aMargin = -leftBearing;
1.1891 + }
1.1892 + if (rightBearing < 0)
1.1893 + {
1.1894 + aClipRect2.iBr.iY -= rightBearing;
1.1895 + }
1.1896 + }
1.1897 + }
1.1898 +
1.1899 +
1.1900 +/**
1.1901 +Draws text vertically, clipped to a specified rectangle, using a baseline offset, alignment and margin.
1.1902 +
1.1903 +@param aText The text string to be drawn.
1.1904 +@param aParam Parameters used in drawing text.
1.1905 +@param aClipFillRect The clipping rectangle (this rect will also be filled before text is plotted).
1.1906 +@param aBaselineOffset Number of pixels to offset the baseline by.
1.1907 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1908 +@param aVert Vertical alignment of the text relative to the specified rectangle.
1.1909 +@param aMargin Offset of the text from the position within the rectangle, using the specified alignment.
1.1910 +
1.1911 +@panic DGDI 7, if the rendering context has not been activated.
1.1912 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush style is neither ENullBrush
1.1913 + nor ESolidBrush.
1.1914 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1915 +*/
1.1916 +EXPORT_C void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect& aClipFillRect, TInt aBaselineOffset,
1.1917 + TBool aUp, DirectGdi::TTextAlign aVert, TInt aMargin)
1.1918 + {
1.1919 + GRAPHICS_TRACE("CDirectGdiContext::DrawTextVertical");
1.1920 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1921 + // This next check covers both bitmap and open fonts
1.1922 + const CBitmapFont* bitmapFont = iFont.Address();
1.1923 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.1924 + DrawTextVertical(aText, aParam, NULL, &aClipFillRect, &aClipFillRect, aBaselineOffset, -1, aUp, aVert, aMargin);
1.1925 + }
1.1926 +
1.1927 +
1.1928 +/**
1.1929 +Draws text vertically, clipped to a specified rectangle, using a baseline offset, alignment and margin.
1.1930 +
1.1931 +@param aText The text string to be drawn.
1.1932 +@param aParam Parameters used in drawing text.
1.1933 +@param aClipFillRect The clipping rectangle (this rect will also be filled before text is plotted).
1.1934 +@param aBaselineOffset Number of pixels to offset the baseline by.
1.1935 +@param aTextWidth Number of pixels to clip the text to.
1.1936 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1937 +@param aVert Vertical alignment of the text relative to the specified rectangle.
1.1938 +@param aMargin Offset of the text from the position within the rectangle, using the specified alignment.
1.1939 +
1.1940 +@panic DGDI 7, if the rendering context has not been activated.
1.1941 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush style is neither ENullBrush
1.1942 + nor ESolidBrush.
1.1943 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1944 +*/
1.1945 +EXPORT_C void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect& aClipFillRect, TInt aBaselineOffset,
1.1946 + TInt aTextWidth, TBool aUp, DirectGdi::TTextAlign aVert, TInt aMargin)
1.1947 + {
1.1948 + GRAPHICS_TRACE("CDirectGdiContext::DrawTextVertical");
1.1949 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.1950 + // This next check covers both bitmap and open fonts
1.1951 + const CBitmapFont* bitmapFont = iFont.Address();
1.1952 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.1953 + DrawTextVertical(aText, aParam, NULL, &aClipFillRect, &aClipFillRect, aBaselineOffset, aTextWidth, aUp, aVert, aMargin);
1.1954 + }
1.1955 +
1.1956 +
1.1957 +/**
1.1958 +The private general DrawTextVertical() routine that implements all the others.
1.1959 +Two clipping rectangles received from different routines. The final rectangle will be calculated as intersection
1.1960 +of first and second clipping rectangle. If aClipRect2 is empty, the error state is set to KErrArgument.
1.1961 +
1.1962 +@param aText The text string to be drawn.
1.1963 +@param aParam Parameters used in drawing text.
1.1964 +@param aClipRect1 Pointer to first clipping rectangle.
1.1965 +@param aClipRect2 Pointer to second clipping rectangle.
1.1966 +@param aFillRect Pointer to rectangle to be filled before text plotting.
1.1967 +@param aBaselineOffset Number of pixels to offset the baseline by.
1.1968 +@param aTextWidth Number of pixels to clip the text to. If negative, the text will not be clipped
1.1969 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.1970 +@param aVert Vertical alignment of the text relative to the specified rectangle.
1.1971 +@param aMargin Offset of the text from the position within the rectangle, using the specified alignment.
1.1972 +
1.1973 +@panic DGDI 10, if the active font is an outline and/or shadow font and the brush style is neither ENullBrush
1.1974 + nor ESolidBrush.
1.1975 +@panic DGDI 11, if a font has not been set prior to calling DrawText().
1.1976 +@panic DGDI 22, if aClipRect2 is NULL.
1.1977 +*/
1.1978 +void CDirectGdiContext::DrawTextVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, const TRect* aClipRect1, const TRect* aClipRect2, const TRect* aFillRect,
1.1979 + TInt aBaselineOffset, TInt aTextWidth, TBool aUp, DirectGdi::TTextAlign aVert, TInt aMargin)
1.1980 + {
1.1981 + GRAPHICS_ASSERT_ALWAYS(aClipRect2, EDirectGdiPanicBadParameter);
1.1982 +
1.1983 + TRect clipRect2 = *aClipRect2;
1.1984 + clipRect2.Move(iOrigin);
1.1985 +
1.1986 + TRect clipRect(clipRect2);
1.1987 + if (aClipRect1 != NULL)
1.1988 + {
1.1989 + if(aClipRect1->IsEmpty())
1.1990 + {
1.1991 + iDriver.SetError(KErrArgument);
1.1992 + return;
1.1993 + }
1.1994 + TRect clipRect1 = *aClipRect1;
1.1995 + clipRect1.Move(iOrigin);
1.1996 + clipRect.Intersection(clipRect1);
1.1997 + }
1.1998 +
1.1999 + if ((aFillRect != NULL) && (iBrushStyle != ENullBrush))
1.2000 + {
1.2001 + // fill the box if necessary
1.2002 + TRect fillBox = *aFillRect;
1.2003 + fillBox.Move(iOrigin);
1.2004 + if (fillBox.Intersects(clipRect))
1.2005 + {
1.2006 + fillBox.Intersection(clipRect);
1.2007 + iEngine->SetPenStyle(DirectGdi::ENullPen); // Fill box, don't outline it
1.2008 + iEngine->DrawRect(*aFillRect);
1.2009 + iEngine->SetPenStyle(iPenStyle); // Put the pen style back
1.2010 + }
1.2011 + }
1.2012 + if (!aText.Length())
1.2013 + {
1.2014 + return;
1.2015 + }
1.2016 + if (aClipRect2->IsEmpty())
1.2017 + {
1.2018 + iDriver.SetError(KErrArgument);
1.2019 + return;
1.2020 + }
1.2021 +
1.2022 + const CBitmapFont* bitmapFont = iFont.Address();
1.2023 + GRAPHICS_ASSERT_ALWAYS(bitmapFont != 0, EDirectGdiPanicNoFontSelected);
1.2024 +
1.2025 + CFont::TMeasureTextInput input;
1.2026 + //CFont::TMeasureTextOutput
1.2027 + if (aParam)
1.2028 + {
1.2029 + GRAPHICS_ASSERT_ALWAYS(aParam->iStart < aParam->iEnd ,EDirectGdiPanicBadParameter);
1.2030 + input.iStartInputChar = aParam->iStart;
1.2031 + input.iEndInputChar = Min(aText.Length(),aParam->iEnd);
1.2032 + }
1.2033 + TInt width = iFont.MeasureText(aText,&input);
1.2034 + TOpenFontMetrics metrics;
1.2035 + iFont.GetFontMetrics(metrics);
1.2036 +
1.2037 + if (aTextWidth < 0)
1.2038 + {
1.2039 + aTextWidth = width;
1.2040 + }
1.2041 + TPoint coords;
1.2042 + coords.iX = clipRect2.iTl.iX;
1.2043 + TInt directionalMultiplier = aUp ? -1 : 1;
1.2044 + coords.iY = aUp ? clipRect2.iBr.iY - 1 : clipRect2.iTl.iY;
1.2045 + //
1.2046 + // iX calculation, for example: ascent(a)=18 descent(d)=2 size=boxwidth=fontheight(h)=20 baseline=ascent
1.2047 + // pre: iX = 0
1.2048 + //
1.2049 + // hhhhhhhhhhhhhhhhhhhh
1.2050 + // 01234567890123456789
1.2051 + // aaaaaaaaaaaaaaaaaadd aUp=ETrue
1.2052 + // ^
1.2053 + // iX = 18 (baseline)
1.2054 + //
1.2055 + // ddaaaaaaaaaaaaaaaaaa aUp=EFalse
1.2056 + // ^
1.2057 + // iX = 1 (instead of 2 ie 20-18-1 which is boxwidth-baseline-1)
1.2058 + //
1.2059 + coords.iX += aUp ? aBaselineOffset : clipRect2.Width() - aBaselineOffset - 1;
1.2060 + switch (aVert)
1.2061 + {
1.2062 + case DirectGdi::ELeft:
1.2063 + coords.iY += aMargin * directionalMultiplier;
1.2064 + break;
1.2065 + case DirectGdi::ECenter:
1.2066 + coords.iY += (((clipRect2.iBr.iY - clipRect2.iTl.iY - aTextWidth) >> 1) + aMargin) * directionalMultiplier;
1.2067 + break;
1.2068 + case DirectGdi::ERight:
1.2069 + coords.iY += (clipRect2.iBr.iY - clipRect2.iTl.iY - aTextWidth - aMargin) * directionalMultiplier;
1.2070 + break;
1.2071 + default:
1.2072 + iDriver.SetError(KErrArgument);
1.2073 + return;
1.2074 + }
1.2075 + iLastPrintPosition = coords;
1.2076 + coords.iX += bitmapFont->iAlgStyle.iBaselineOffsetInPixels * directionalMultiplier;
1.2077 + TInt prewidth = width + iCharJustExcess + iWordJustExcess;
1.2078 + iLastPrintPosition.iY -= aUp ? prewidth - 1 : -prewidth;
1.2079 + if (clipRect.IsEmpty() || !width)
1.2080 + {
1.2081 + if (iAutoUpdateJustification)
1.2082 + {
1.2083 + UpdateJustificationVertical(aText, aParam, aUp);
1.2084 + }
1.2085 + return;
1.2086 + }
1.2087 +
1.2088 + /*
1.2089 + Set up the parameter block for character positioning.
1.2090 + Draw left to right, because although the text is being drawn vertically,
1.2091 + it is done by rotating the baseline 90 degrees and drawing in the ordinary way, not by drawing
1.2092 + the characters in their normal orientation but in a vertical column.
1.2093 + */
1.2094 + CFont::TPositionParam param;
1.2095 + param.iText.Set(aText);
1.2096 + param.iPen = coords;
1.2097 + TInt endDraw = aText.Length();
1.2098 + if (aParam)
1.2099 + {
1.2100 + param.iPosInText = aParam->iStart;
1.2101 + endDraw = Min(aText.Length(),aParam->iEnd);
1.2102 + }
1.2103 + else
1.2104 + {
1.2105 + param.iPosInText = 0;
1.2106 + }
1.2107 +
1.2108 + // Draw the text.
1.2109 + DoDrawTextVertical(param, aUp, endDraw, clipRect);
1.2110 + if(iAutoUpdateJustification)
1.2111 + {
1.2112 + UpdateJustificationVertical(aText, aParam, aUp);
1.2113 + }
1.2114 + }
1.2115 +
1.2116 +
1.2117 +/**
1.2118 +Draws vertical text within the clipping area
1.2119 +
1.2120 +@param aParam Defines glyph code, ligature creation and diacritic placement
1.2121 +@param aUp ETrue, text is rotated 90 degrees anti-clockwise; EFalse, text is rotated 90 degrees clockwise.
1.2122 +@param aEnd The end position within the text descriptor to draw.
1.2123 +@param aClipRect The clipping rectangle.
1.2124 +
1.2125 +@pre iFont is a valid CFont.
1.2126 +
1.2127 +@panic DGDI 13, if the active font is an outline and/or shadow font and the brush style is neither ENullBrush
1.2128 + nor ESolidBrush.
1.2129 +*/
1.2130 +void CDirectGdiContext::DoDrawTextVertical(CFont::TPositionParam& aParam, TBool aUp, const TInt aEnd, TRect& aClipRect)
1.2131 + {
1.2132 + const CBitmapFont* bitmapFont = iFont.Address();
1.2133 + if ((bitmapFont->GlyphBitmapType() == EFourColourBlendGlyphBitmap) && !((iBrushStyle == DirectGdi::ENullBrush) || (iBrushStyle == DirectGdi::ESolidBrush)))
1.2134 + {
1.2135 + //For future compatibility it is better if brush style of ENullBrush or ESolidBrush is used
1.2136 + //when drawing outline and shadow fonts.
1.2137 + GRAPHICS_PANIC_ALWAYS(EDirectGdiPanicInvalidBrushStyle);
1.2138 + }
1.2139 +
1.2140 + TPoint startPen = aParam.iPen;
1.2141 + TInt charClipping = aClipRect.iTl.iY;
1.2142 + TInt underlineTop = 0;
1.2143 + TInt underlineBottom = 0;
1.2144 +
1.2145 + //for linked fonts need an adjustment to the underline postion
1.2146 + TInt underlineStrikeoutOffset = BaselineCorrection();
1.2147 +
1.2148 + if (iUnderline == DirectGdi::EUnderlineOn)
1.2149 + {
1.2150 + GetUnderlineMetrics(underlineTop, underlineBottom);
1.2151 + underlineTop+=underlineStrikeoutOffset;
1.2152 + underlineBottom+=underlineStrikeoutOffset;
1.2153 + }
1.2154 + TInt strikeTop = 0;
1.2155 + TInt strikeBottom = 0;
1.2156 + if (iStrikethrough == DirectGdi::EStrikethroughOn)
1.2157 + {
1.2158 + GetStrikethroughMetrics(strikeTop, strikeBottom);
1.2159 + strikeTop+=underlineStrikeoutOffset;
1.2160 + strikeBottom+=underlineStrikeoutOffset;
1.2161 + }
1.2162 +
1.2163 + const DirectGdi::TGraphicsRotation rotation = aUp ? DirectGdi::EGraphicsRotation270 : DirectGdi::EGraphicsRotation90;
1.2164 + iEngine->BeginDrawGlyph();
1.2165 + RShapeInfo shapeInfo;
1.2166 + while (aParam.iPosInText < aEnd)
1.2167 + {
1.2168 + TPoint startPen2 = aParam.iPen;
1.2169 + if (!iFont.GetCharacterPosition2(aParam, shapeInfo))
1.2170 + {
1.2171 + continue;
1.2172 + }
1.2173 + Rotate(aParam.iPen, startPen2, aUp);
1.2174 + TInt adjustment = 0;
1.2175 + if(iCharJustExcess && (iCharJustNum > 0)) // character clipping/justification
1.2176 + {
1.2177 + adjustment = CGraphicsContext::JustificationInPixels(iCharJustExcess, iCharJustNum);
1.2178 + if (adjustment < 0)
1.2179 + {
1.2180 + aClipRect.iTl.iY = aParam.iPen.iY + (aUp ? -adjustment : adjustment);
1.2181 + }
1.2182 + }
1.2183 +
1.2184 + CFont::TPositionParam::TOutput* output = aParam.iOutput;
1.2185 + for (TInt i = 0; i < aParam.iOutputGlyphs; i++, output++)
1.2186 + {
1.2187 + Rotate(output->iBounds.iTl, startPen2, aUp);
1.2188 +
1.2189 + //get the character metrics for the glyph type
1.2190 + TOpenFontCharMetrics characterParams;
1.2191 + const TUint8* bitmap;
1.2192 + TSize size;
1.2193 + //note may now be using a glyph code, and not a character
1.2194 + iFont.GetCharacterData(aParam.iOutput[i].iCode,characterParams,bitmap,size);
1.2195 + TGlyphBitmapType glyphType = characterParams.GlyphType();
1.2196 +
1.2197 + switch (glyphType)
1.2198 + {
1.2199 + case EAntiAliasedGlyphBitmap:
1.2200 + case EFourColourBlendGlyphBitmap:
1.2201 + case EDefaultGlyphBitmap:
1.2202 + iEngine->DrawGlyph(output->iBounds.iTl, output->iCode, output->iBitmap, glyphType, output->iBitmapSize, aClipRect, rotation);
1.2203 + break;
1.2204 +
1.2205 + default:
1.2206 + //if the outline or shadow is not specified for the character, then use the font setting
1.2207 + iEngine->DrawGlyph(output->iBounds.iTl, output->iCode, output->iBitmap, bitmapFont->GlyphBitmapType(), output->iBitmapSize, aClipRect, rotation);
1.2208 + break;
1.2209 + }
1.2210 + }
1.2211 +
1.2212 + aClipRect.iTl.iY = charClipping;
1.2213 + if (adjustment)
1.2214 + {
1.2215 + aParam.iPen.iY += aUp ? -adjustment : adjustment;
1.2216 + }
1.2217 + if ((iWordJustExcess > 0) && (iWordJustNum > 0) && (aParam.iOutput[0].iCode == 0x0020)) // word justification
1.2218 + {
1.2219 + adjustment = CGraphicsContext::JustificationInPixels(iWordJustExcess, iWordJustNum);
1.2220 + aParam.iPen.iY += aUp ? -adjustment : adjustment;
1.2221 + }
1.2222 + }
1.2223 + iEngine->EndDrawGlyph();
1.2224 + if (shapeInfo.IsOpen())
1.2225 + {
1.2226 + shapeInfo.Close();
1.2227 + }
1.2228 +
1.2229 + if (iUnderline == DirectGdi::EUnderlineOn)
1.2230 + {
1.2231 + TRect underlineRect; // underline
1.2232 + if (aUp)
1.2233 + {
1.2234 + underlineRect.SetRect(startPen.iX + underlineTop, aParam.iPen.iY, startPen.iX + underlineBottom, startPen.iY + 1);
1.2235 + underlineRect.iTl.iY = underlineRect.iBr.iY - underlineRect.Height();
1.2236 + }
1.2237 + else
1.2238 + {
1.2239 + underlineRect.SetRect(startPen.iX - underlineBottom, startPen.iY, startPen.iX - underlineTop, aParam.iPen.iY);
1.2240 + underlineRect.iBr.iY = underlineRect.iTl.iY + underlineRect.Height();
1.2241 + underlineRect.iTl.iX++; // adjust for rect not including last line
1.2242 + underlineRect.iBr.iX++;
1.2243 + }
1.2244 + FillRect(underlineRect, iPenColor, aClipRect);
1.2245 + }
1.2246 +
1.2247 + if (iStrikethrough == DirectGdi::EStrikethroughOn)
1.2248 + {
1.2249 + TRect strikethroughRect; // strikethrough
1.2250 + if (aUp)
1.2251 + {
1.2252 + strikethroughRect.SetRect(startPen.iX + strikeTop, aParam.iPen.iY, startPen.iX + strikeBottom, startPen.iY + 1);
1.2253 + strikethroughRect.iTl.iY = strikethroughRect.iBr.iY - strikethroughRect.Height();
1.2254 + }
1.2255 + else
1.2256 + {
1.2257 + strikethroughRect.SetRect(startPen.iX - strikeBottom, startPen.iY, startPen.iX - strikeTop, aParam.iPen.iY);
1.2258 + strikethroughRect.iBr.iY = strikethroughRect.iTl.iY + strikethroughRect.Height();
1.2259 + strikethroughRect.iTl.iX++;
1.2260 + strikethroughRect.iBr.iX++;
1.2261 + }
1.2262 + FillRect(strikethroughRect, iPenColor, aClipRect);
1.2263 + }
1.2264 + }
1.2265 +
1.2266 +
1.2267 +/**
1.2268 +Transform a vector, defined by a point relative to an origin, from left-to-right to up or down.
1.2269 +
1.2270 +@param aPoint A point relative to the origin aOrigin.
1.2271 +@param aOrigin The origin to use when transforming the point aPoint.
1.2272 +@param aUp If ETrue, then transform the point from left-right to up, otherwise transform from
1.2273 +left-right to down.
1.2274 +*/
1.2275 +void CDirectGdiContext::Rotate(TPoint& aPoint, const TPoint& aOrigin, TBool aUp)
1.2276 + {
1.2277 + TInt dx = aPoint.iX - aOrigin.iX;
1.2278 + TInt dy = aPoint.iY - aOrigin.iY;
1.2279 + if (aUp)
1.2280 + {
1.2281 + aPoint.iX = aOrigin.iX + dy;
1.2282 + aPoint.iY = aOrigin.iY - dx;
1.2283 + }
1.2284 + else
1.2285 + {
1.2286 + aPoint.iX = aOrigin.iX - dy;
1.2287 + aPoint.iY = aOrigin.iY + dx;
1.2288 + }
1.2289 + }
1.2290 +
1.2291 +
1.2292 +/**
1.2293 +Can be used to find out the top and bottom of an underline for the active font.
1.2294 +This allows correct calculation of the area required in which to draw text with underline.
1.2295 +
1.2296 +@param aTop The top of the underline position.
1.2297 +@param aBottom The bottom of the underline position.
1.2298 +*/
1.2299 +void CDirectGdiContext::GetUnderlineMetrics(TInt& aTop, TInt& aBottom)
1.2300 + {
1.2301 + const TInt width = Max((iFont.HeightInPixels() / 10), 1);
1.2302 + aTop = 1 + (width >> 1);
1.2303 + aBottom = aTop + width;
1.2304 + }
1.2305 +
1.2306 +
1.2307 +/**
1.2308 +Get the top and bottom of a strikethrough line for the current font, relative to the baseline.
1.2309 +
1.2310 +@param aTop The top of the strikethrough position.
1.2311 +@param aBottom The bottom of the strikethrough position.
1.2312 +*/
1.2313 +void CDirectGdiContext::GetStrikethroughMetrics(TInt& aTop, TInt& aBottom)
1.2314 + {
1.2315 + aTop = -(iFont.AscentInPixels() * 5/12) - 1;
1.2316 + aBottom = aTop + Max((iFont.HeightInPixels() / 10), 1);
1.2317 + }
1.2318 +
1.2319 +
1.2320 +/**
1.2321 +Moves the internal drawing position relative to the co-ordinate origin, without drawing a line.
1.2322 +A subsequent call to DrawLineTo() or DrawLineBy() will then use the new internal drawing position
1.2323 +as the start point for the line drawn.
1.2324 +
1.2325 +The operations DrawLine(), DrawLineTo(), DrawLineBy() and DrawPolyline() also change the internal drawing
1.2326 +position to the last point of the drawn line(s). The internal drawing position is set to the co-ordinate
1.2327 +origin if no drawing or moving operations have yet taken place.
1.2328 +
1.2329 +@see CDirectGdiContext::MoveBy(const TPoint&)
1.2330 +
1.2331 +@param aPoint The point to move the internal drawing position to.
1.2332 +
1.2333 +@pre The rendering target has been activated.
1.2334 +@post Request to move the internal drawing position to the specified point has been accepted.
1.2335 + There is no guarantee that the request has been processed when the method returns.
1.2336 +
1.2337 +@panic DGDI 7, if the rendering context has not been activated.
1.2338 +*/
1.2339 +EXPORT_C void CDirectGdiContext::MoveTo(const TPoint& aPoint)
1.2340 + {
1.2341 + GRAPHICS_TRACE2("CDirectGdiContext::MoveTo(%d,%d)", aPoint.iX, aPoint.iY);
1.2342 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.2343 + iEngine->MoveTo(aPoint);
1.2344 + }
1.2345 +
1.2346 +
1.2347 +/**
1.2348 +Moves the internal drawing position by a vector, relative to the current position, without drawing a line.
1.2349 +A subsequent call to DrawLineTo() or DrawLineBy() will then use the new internal drawing position
1.2350 +as the start point for the line drawn.
1.2351 +
1.2352 +The operations DrawLine(), DrawLineTo(), DrawLineBy() and DrawPolyline() also change the internal drawing
1.2353 +position to the last point of the drawn line(s). The internal drawing position is set to the co-ordinate
1.2354 +origin if no drawing or moving operations have yet taken place.
1.2355 +
1.2356 +@see CDirectGdiContext::MoveTo(const TPoint&)
1.2357 +
1.2358 +@param aVector The vector to move the internal position by.
1.2359 +
1.2360 +@pre The rendering target has been activated.
1.2361 +@post Request to move the internal drawing position by a vector has been accepted.
1.2362 + There is no guarantee that the request has been processed when the method returns.
1.2363 +@panic DGDI 7, if the rendering context has not been activated.
1.2364 +*/
1.2365 +EXPORT_C void CDirectGdiContext::MoveBy(const TPoint& aVector)
1.2366 + {
1.2367 + GRAPHICS_TRACE2("CDirectGdiContext::MoveBy(%d,%d)", aVector.iX, aVector.iY);
1.2368 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.2369 + if (aVector != TPoint(0,0))
1.2370 + {
1.2371 + iEngine->MoveBy(aVector);
1.2372 + }
1.2373 + }
1.2374 +
1.2375 +
1.2376 +/**
1.2377 +Draws a point at given location using current pen colour and size.
1.2378 +If the pen size is greater than 1x1 pixel, a filled circle/ellipse with current pen
1.2379 +colour should be drawn with the given position as the centre.
1.2380 +
1.2381 +@param aPoint The position to plot.
1.2382 +
1.2383 +@pre The rendering target has been activated.
1.2384 +@post Request to draw a point or filled circle/ellipse has been accepted.
1.2385 + There is no guarantee that the request has been processed when the method returns.
1.2386 +
1.2387 +@panic DGDI 7, if the rendering context has not been activated.
1.2388 +
1.2389 +@see CDirectGdiContext::SetPenSize(const TSize&)
1.2390 +@see CDirectGdiContext::SetPenColor(TRgb)
1.2391 +@see CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode)
1.2392 +*/
1.2393 +EXPORT_C void CDirectGdiContext::Plot(const TPoint& aPoint)
1.2394 + {
1.2395 + GRAPHICS_TRACE2("CDirectGdiContext::Plot(%d,%d)", aPoint.iX, aPoint.iY);
1.2396 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.2397 + if (iPenStyle == DirectGdi::ENullPen || iPenSize.iWidth == 0 || iPenSize.iHeight == 0)
1.2398 + {
1.2399 + return;
1.2400 + }
1.2401 + iEngine->Plot(aPoint);
1.2402 + }
1.2403 +
1.2404 +
1.2405 +/**
1.2406 +Resets drawing state to its default settings. This operation does not unbind the current target.
1.2407 +@pre None.
1.2408 +@post The drawing state is reset to default values. Subsequent drawing will use the default settings until they are changed to different values.
1.2409 +*/
1.2410 +EXPORT_C void CDirectGdiContext::Reset()
1.2411 + {
1.2412 + GRAPHICS_TRACE("CDirectGdiContext::Reset");
1.2413 + iEngine->Reset();
1.2414 +
1.2415 + // Explicit calls are made to the engine to apply the defaults.
1.2416 + // Set() methods should generally not be used as they perform unnecessary checks, and may also
1.2417 + // block the call to the engine if the value being set is the same as the constructor's default.
1.2418 + // Clipping regions.
1.2419 + iClippingRegion.Clear();
1.2420 + iEngine->ResetClippingRegion();
1.2421 + // Origin.
1.2422 + iOrigin.SetXY(0,0);
1.2423 + iEngine->SetOrigin(iOrigin);
1.2424 + // Font.
1.2425 + iEngine->ResetFont();
1.2426 + iFont.Reset();
1.2427 + // Text.
1.2428 + iLastPrintPosition.SetXY(0,0);
1.2429 + iAutoUpdateJustification = ETrue;
1.2430 + SetCharJustification(0, 0);
1.2431 + SetWordJustification(0, 0);
1.2432 + SetStrikethroughStyle(DirectGdi::EStrikethroughOff);
1.2433 + SetUnderlineStyle(DirectGdi::EUnderlineOff);
1.2434 + // Pen colour.
1.2435 + iPenColor = KRgbBlack;
1.2436 + iEngine->SetPenColor(iPenColor);
1.2437 + // Pen size.
1.2438 + iPenSize = TSize(1,1);
1.2439 + iEngine->SetPenSize(iPenSize);
1.2440 + // Pen style.
1.2441 + iPenStyle = DirectGdi::ESolidPen;
1.2442 + iEngine->SetPenStyle(iPenStyle);
1.2443 + // Draw mode.
1.2444 + iDrawMode = DirectGdi::EDrawModePEN;
1.2445 + iEngine->SetDrawMode(iDrawMode);
1.2446 + // Text shadow colour.
1.2447 + iTextShadowColor = KRgbGray;
1.2448 + iEngine->SetTextShadowColor(iTextShadowColor);
1.2449 + // Brush colour.
1.2450 + iBrushColor = KRgbWhite;
1.2451 + iEngine->SetBrushColor(iBrushColor);
1.2452 + // Brush style.
1.2453 + iBrushStyle = DirectGdi::ENullBrush;
1.2454 + iEngine->SetBrushStyle(iBrushStyle);
1.2455 + // Brush pattern.
1.2456 + CleanUpBrushPattern();
1.2457 + iEngine->ResetBrushPattern();
1.2458 + iBrushPatternUsed = EFalse;
1.2459 + // Brush origin.
1.2460 + iBrushOrigin.SetXY(0,0);
1.2461 + iEngine->SetBrushOrigin(iBrushOrigin);
1.2462 + // Internal drawing position.
1.2463 + iEngine->MoveTo(TPoint(0,0));
1.2464 + }
1.2465 +
1.2466 +
1.2467 +/**
1.2468 +Sets the colour for clearing, filling the area of shapes and the background of text boxes.
1.2469 +
1.2470 +The default brush colour is white. However, the default brush style is ENullBrush, so when drawing
1.2471 +to a target the default appears to be the target's background colour.
1.2472 +
1.2473 +@see CDirectGdiContext::Clear()
1.2474 +@see CDirectGdiContext::Clear(const TRect&)
1.2475 +@see CDirectGdiContext::DrawRect()
1.2476 +@see CDirectGdiContext::DrawRoundRect()
1.2477 +@see CDirectGdiContext::DrawPolygon()
1.2478 +@see CDirectGdiContext::DrawPie()
1.2479 +
1.2480 +@param aColor Brush colour.
1.2481 +
1.2482 +@pre None.
1.2483 +@post The new brush colour will be used on subsequent drawing operations if a brush style making
1.2484 + use of the brush colour is used. The new brush colour remains in effect until another SetBrushColor()
1.2485 + with a different parameter is called.
1.2486 +*/
1.2487 +EXPORT_C void CDirectGdiContext::SetBrushColor(const TRgb& aColor)
1.2488 + {
1.2489 + GRAPHICS_TRACE1("CDirectGdiContext::SetBrushColor(%d)", aColor.Internal());
1.2490 + if (aColor != iBrushColor)
1.2491 + {
1.2492 + iBrushColor = aColor;
1.2493 + iEngine->SetBrushColor(iBrushColor);
1.2494 + }
1.2495 + }
1.2496 +
1.2497 +
1.2498 +/**
1.2499 +Sets the brush pattern origin which specifies the start of a pattern tile.
1.2500 +Shapes can be considered as a view port into a continuous pattern tile covering the entire
1.2501 +area of rendering target. The default origin is TPoint(0,0).
1.2502 +
1.2503 +@see CDirectGdiContext::SetBrushPattern()
1.2504 +
1.2505 +@param aOrigin An origin point for the brush. The coordinates are relative
1.2506 + to the rectangle to fill, i.e. specify TPoint(0,0) to align the pattern flush with
1.2507 + the top and left hand sides of the rectangle.
1.2508 +
1.2509 +@pre None.
1.2510 +@post New brush origin will be used when filling an area with a pattern is used on
1.2511 + subsequent drawing operations. It remains in effect until another SetBrushOrigin()
1.2512 + with a different parameter is called.
1.2513 +*/
1.2514 +EXPORT_C void CDirectGdiContext::SetBrushOrigin(const TPoint& aOrigin)
1.2515 + {
1.2516 + GRAPHICS_TRACE2("CDirectGdiContext::SetBrushOrigin(%d,%d)", aOrigin.iX, aOrigin.iY);
1.2517 + if (aOrigin != iBrushOrigin)
1.2518 + {
1.2519 + iBrushOrigin = aOrigin;
1.2520 + iEngine->SetBrushOrigin(iBrushOrigin);
1.2521 + }
1.2522 + }
1.2523 +
1.2524 +
1.2525 +/**
1.2526 +Sets the brush style used when filling the area of shapes and the background of text boxes.
1.2527 +Use ENullBrush to draw the outline of a fillable shape on its own, without filling.
1.2528 +
1.2529 +The error state is set to KErrArgument if aBrushStyle is an invalid brush style.
1.2530 +
1.2531 +@see DirectGdi::TBrushStyle
1.2532 +
1.2533 +@param aBrushStyle The brush style to set.
1.2534 +
1.2535 +@pre If aBrushStyle is EPatternedBrush, a pattern must have been set first using SetBrushPattern().
1.2536 +@post New brush style will be used for subsequent drawing operations, and remains in effect
1.2537 + until another SetBrushStyle() with a different parameter is called.
1.2538 +@panic DGDI 9, if aBrushStyle is EPatternedBrush but no brush pattern has successfully been set.
1.2539 +*/
1.2540 +EXPORT_C void CDirectGdiContext::SetBrushStyle(DirectGdi::TBrushStyle aBrushStyle)
1.2541 + {
1.2542 + GRAPHICS_TRACE1("CDirectGdiContext::SetBrushStyle(%d)", aBrushStyle);
1.2543 + if (aBrushStyle < DirectGdi::ENullBrush || aBrushStyle > DirectGdi::EDiamondCrossHatchBrush)
1.2544 + {
1.2545 + iDriver.SetError(KErrArgument);
1.2546 + return;
1.2547 + }
1.2548 +
1.2549 + if (aBrushStyle != iBrushStyle)
1.2550 + {
1.2551 + GRAPHICS_ASSERT_ALWAYS(aBrushStyle != DirectGdi::EPatternedBrush || iBrushPatternUsed, EDirectGdiPanicBrushPatternNotSet);
1.2552 + iBrushStyle = aBrushStyle;
1.2553 + iEngine->SetBrushStyle(iBrushStyle);
1.2554 + }
1.2555 + }
1.2556 +
1.2557 +
1.2558 +/**
1.2559 +Sets a clipping region which will be used to clip subsequent rendering operations on the current target.
1.2560 +This operation is non-additive, any previous clipping region setting is replaced by the new one. A clipping
1.2561 +region can contain one or more rectangles and is specified in absolute values in the target coordinate system.
1.2562 +By default (when a target is activated for the first time) no clipping region is set and any drawing
1.2563 +operations will be clipped automatically to the full area of the rendering target.
1.2564 +
1.2565 +In the event of a failure, the error state is set to KErrArgument if the given region is invalid or not
1.2566 +fully contained within the area of target, otherwise one of the system-wide error codes.
1.2567 +
1.2568 +@see CDirectGdiContext::ResetClippingRegion()
1.2569 +
1.2570 +@param aRegion The new clipping region.
1.2571 +
1.2572 +@pre Region is not empty and is fully contained within the full area of the target.
1.2573 +@post Subsequent rendering operations will be clipped to the given region if there is no error
1.2574 + while performing the operation, otherwise previous clipping region settings will be retained.
1.2575 +*/
1.2576 +EXPORT_C void CDirectGdiContext::SetClippingRegion(const TRegion& aRegion)
1.2577 + {
1.2578 + GRAPHICS_TRACE("CDirectGdiContext::SetClippingRegion");
1.2579 + if (aRegion.CheckError())
1.2580 + {
1.2581 + iDriver.SetError(KErrArgument);
1.2582 + return;
1.2583 + }
1.2584 + iClippingRegion.Copy(aRegion);
1.2585 + if (iClippingRegion.CheckError())
1.2586 + {
1.2587 + iDriver.SetError(KErrNoMemory);
1.2588 + return;
1.2589 + }
1.2590 + iEngine->SetClippingRegion(iClippingRegion);
1.2591 + }
1.2592 +
1.2593 +
1.2594 +/**
1.2595 +Sets the drawing mode which will affect the way pen and brush colour are used in rendering operations.
1.2596 +The default drawing mode is EDrawModePEN.
1.2597 +
1.2598 +The error state is set to KErrArgument if aDrawMode is an invalid draw mode.
1.2599 +
1.2600 +@see DirectGdi::TDrawMode
1.2601 +
1.2602 +@param aDrawMode The drawing mode.
1.2603 +
1.2604 +@pre None.
1.2605 +@post The new drawing mode will be applied to subsequent rendering operations, and remains in effect
1.2606 + until another SetDrawMode() with a different parameter is called.
1.2607 +*/
1.2608 +EXPORT_C void CDirectGdiContext::SetDrawMode(DirectGdi::TDrawMode aDrawMode)
1.2609 + {
1.2610 + GRAPHICS_TRACE1("CDirectGdiContext::SetDrawMode(%d)", aDrawMode);
1.2611 + if (aDrawMode != DirectGdi::EDrawModePEN && aDrawMode != DirectGdi::EDrawModeWriteAlpha)
1.2612 + {
1.2613 + iDriver.SetError(KErrArgument);
1.2614 + return;
1.2615 + }
1.2616 +
1.2617 + if (iDrawMode != aDrawMode)
1.2618 + {
1.2619 + iDrawMode = aDrawMode;
1.2620 + iEngine->SetDrawMode(iDrawMode);
1.2621 + }
1.2622 + }
1.2623 +
1.2624 +
1.2625 +/**
1.2626 +Sets the origin of the drawing engine coordinate system. By default this is TPoint(0,0) and
1.2627 +coincides with the origin of the target coordinate system which is at the top-left corner of
1.2628 +the full area of target. The X value increases from left to right, and Y value increases from
1.2629 +top to bottom. Integer values are used to represent the position within the coordinate system.
1.2630 +
1.2631 +All drawing operations are done relative to the engine’s origin. However, the clipping region
1.2632 +is always specified in absolute coordinate values (using the target coordinate system) and is
1.2633 +not affected by changes to the drawing engine’s coordinate system origin.
1.2634 +
1.2635 +@param aPoint The new origin for the drawing engine’s coordinate system.
1.2636 +
1.2637 +@pre None.
1.2638 +@post The origin of the drawing engine’s coordinate system is moved to the given position.
1.2639 + All subsequent drawing operations will be done relative to the new origin. The new origin remains
1.2640 + in effect until SetOrigin() is called again with a different parameter.
1.2641 +*/
1.2642 +EXPORT_C void CDirectGdiContext::SetOrigin(const TPoint& aPoint)
1.2643 + {
1.2644 + GRAPHICS_TRACE2("CDirectGdiContext::SetOrigin(%d,%d)", aPoint.iX, aPoint.iY);
1.2645 + if (aPoint != iOrigin)
1.2646 + {
1.2647 + iOrigin = aPoint;
1.2648 + iEngine->SetOrigin(iOrigin);
1.2649 + }
1.2650 + }
1.2651 +
1.2652 +
1.2653 +/**
1.2654 +Sets the colour that will be used for drawing lines, outlines of filled shapes and text. The
1.2655 +default pen colour is black. For outline and shadow fonts the alpha value of the pen colour will be
1.2656 +used for blending the font to the destination.
1.2657 +
1.2658 +@see CDirectGdiContext::Plot()
1.2659 +@see CDirectGdiContext::DrawLine()
1.2660 +@see CDirectGdiContext::DrawRoundRect()
1.2661 +@see CDirectGdiContext::DrawRect()
1.2662 +@see CDirectGdiContext::DrawPolyLine()
1.2663 +@see CDirectGdiContext::DrawPolyLineNoEndPoint()
1.2664 +@see CDirectGdiContext::DrawPolygon()
1.2665 +@see CDirectGdiContext::DrawPie()
1.2666 +@see CDirectGdiContext::DrawArc()
1.2667 +@see CDirectGdiContext::DrawText()
1.2668 +
1.2669 +@param aColor The pen colour.
1.2670 +
1.2671 +@pre None.
1.2672 +@post The new pen colour will be used for subsequent drawing of lines, outlines of filled shapes and text.
1.2673 + The new pen colour remains in effect until another SetPenColor() with a different parameter is called.
1.2674 +*/
1.2675 +EXPORT_C void CDirectGdiContext::SetPenColor(const TRgb& aColor)
1.2676 + {
1.2677 + GRAPHICS_TRACE1("CDirectGdiContext::SetPenColor(%d)", aColor.Internal());
1.2678 + if (aColor != iPenColor)
1.2679 + {
1.2680 + iPenColor = aColor;
1.2681 + iEngine->SetPenColor(iPenColor);
1.2682 + }
1.2683 + }
1.2684 +
1.2685 +
1.2686 +/**
1.2687 +Sets the pen or line drawing style.
1.2688 +
1.2689 +The pen style is used to draw lines and outlines of shapes. ENullPen can be used if border or
1.2690 +outlines are not required (when drawing a filled shape). The default pen style is ESolidPen.
1.2691 +
1.2692 +The error state is set to KErrArgument if aPenStyle is an invalid pen style.
1.2693 +
1.2694 +@see CDirectGdiContext::Plot()
1.2695 +@see CDirectGdiContext::DrawLine()
1.2696 +@see CDirectGdiContext::DrawRoundRect()
1.2697 +@see CDirectGdiContext::DrawRect()
1.2698 +@see CDirectGdiContext::DrawPolyLine()
1.2699 +@see CDirectGdiContext::DrawPolyLineNoEndPoint()
1.2700 +@see CDirectGdiContext::DrawPolygon()
1.2701 +@see CDirectGdiContext::DrawPie()
1.2702 +@see CDirectGdiContext::DrawArc()
1.2703 +@see DirectGdi::TPenStyle
1.2704 +
1.2705 +@param aPenStyle The pen style.
1.2706 +
1.2707 +@pre None.
1.2708 +@post The new pen style will be applied for subsequent drawing lines and outlines of filled shapes.
1.2709 + The new pen style remains in effect until another SetPenStyle() with a different parameter is called.
1.2710 +*/
1.2711 +EXPORT_C void CDirectGdiContext::SetPenStyle(DirectGdi::TPenStyle aPenStyle)
1.2712 + {
1.2713 + GRAPHICS_TRACE1("CDirectGdiContext::SetPenStyle(%d)", aPenStyle);
1.2714 + if (aPenStyle < DirectGdi::ENullPen || aPenStyle > DirectGdi::EDotDotDashPen)
1.2715 + {
1.2716 + iDriver.SetError(KErrArgument);
1.2717 + return;
1.2718 + }
1.2719 +
1.2720 + if (aPenStyle != iPenStyle)
1.2721 + {
1.2722 + iPenStyle = aPenStyle;
1.2723 + iEngine->SetPenStyle(iPenStyle);
1.2724 + }
1.2725 + }
1.2726 +
1.2727 +
1.2728 +/**
1.2729 +Sets the pen size for drawing lines or the outlines of a filled shape. The default pen size is 1.
1.2730 +Lines with pen size greater than 1 are drawn with rounded ends that extend beyond the end points
1.2731 +and are always drawn using EDrawModePEN for compatibility reasons.
1.2732 +
1.2733 +The error state is set to KErrArgument if the specified width or height is negative.
1.2734 +
1.2735 +@see CDirectGdiContext::Plot()
1.2736 +@see CDirectGdiContext::DrawLine()
1.2737 +@see CDirectGdiContext::DrawRoundRect()
1.2738 +@see CDirectGdiContext::DrawRect()
1.2739 +@see CDirectGdiContext::DrawPolyLine()
1.2740 +@see CDirectGdiContext::DrawPolyLineNoEndPoint()
1.2741 +@see CDirectGdiContext::DrawPolygon()
1.2742 +@see CDirectGdiContext::DrawPie()
1.2743 +@see CDirectGdiContext::DrawArc()
1.2744 +
1.2745 +@param aSize The pen size.
1.2746 +
1.2747 +@pre None.
1.2748 +@post The new pen size is used for subsequent drawing lines and outlines of filled shapes. The new
1.2749 + pen size remains in effect until another SetPenSize() with a different parameter is called.
1.2750 +*/
1.2751 +EXPORT_C void CDirectGdiContext::SetPenSize(const TSize& aSize)
1.2752 + {
1.2753 + GRAPHICS_TRACE2("CDirectGdiContext::SetPenSize(%d,%d)", aSize.iWidth, aSize.iHeight);
1.2754 + if ((aSize.iWidth < 0) || (aSize.iHeight < 0))
1.2755 + {
1.2756 + iDriver.SetError(KErrArgument);
1.2757 + return;
1.2758 + }
1.2759 +
1.2760 + if (aSize != iPenSize)
1.2761 + {
1.2762 + iPenSize = aSize;
1.2763 + iEngine->SetPenSize(iPenSize);
1.2764 + }
1.2765 + }
1.2766 +
1.2767 +
1.2768 +/**
1.2769 +Sets the colour that will be used for drawing the shadow for shadowed text.
1.2770 +
1.2771 +@param aColor The shadow colour.
1.2772 +
1.2773 +@pre None.
1.2774 +@post The new colour will be used for subsequent drawing of text which has a type EFourColourBlendGlyphBitmap.
1.2775 + The shadow component of the text will be filled with this colour.
1.2776 + The new pen colour remains in effect until another SetTextShadowColor() with a different parameter is called.
1.2777 +*/
1.2778 +EXPORT_C void CDirectGdiContext::SetTextShadowColor(const TRgb& aColor)
1.2779 + {
1.2780 + GRAPHICS_TRACE1("CDirectGdiContext::SetTextShadowColor(%d)", aColor.Internal());
1.2781 + if (aColor != iTextShadowColor)
1.2782 + {
1.2783 + iTextShadowColor = aColor;
1.2784 + iEngine->SetTextShadowColor(aColor);
1.2785 + }
1.2786 + }
1.2787 +
1.2788 +
1.2789 +/**
1.2790 +Sets the character justification.
1.2791 +The function provides a concrete implementation of the pure virtual function CGraphicsContext::SetCharJustification().
1.2792 +The function behaviour is the same as documented (in detail) in that class.
1.2793 +
1.2794 +@param aExcessWidth The excess width (in pixels) to be distributed between the specified number of characters.
1.2795 +@param aNumChars The number of characters involved.
1.2796 +
1.2797 +@see CGraphicsContext::SetCharJustification()
1.2798 +*/
1.2799 +EXPORT_C void CDirectGdiContext::SetCharJustification(TInt aExcessWidth, TInt aNumChars)
1.2800 + {
1.2801 + GRAPHICS_TRACE2("CDirectGdiContext::SetCharJustification(%d,%d)", aExcessWidth, aNumChars);
1.2802 + if (aExcessWidth == 0 || aNumChars <= 0)
1.2803 + {
1.2804 + iCharJustExcess = 0;
1.2805 + iCharJustNum = 0;
1.2806 + }
1.2807 + else
1.2808 + {
1.2809 + iCharJustExcess = aExcessWidth;
1.2810 + iCharJustNum = aNumChars;
1.2811 + }
1.2812 + }
1.2813 +
1.2814 +
1.2815 +/**
1.2816 +Sets the word justification.
1.2817 +The function provides a concrete implementation of the pure virtual function CGraphicsContext::SetWordJustification().
1.2818 +The function behaviour is the same as documented (in detail) in that class.
1.2819 +
1.2820 +@param aExcessWidth The width (in pixels) to be distributed between the specified number of spaces.
1.2821 + It may be positive, in which case the text is stretched, or negative, in which case it is shrunk.
1.2822 +@param aNumGaps The number of word spaces (characters with the code U+0020) over which the change in width is distributed.
1.2823 +
1.2824 +@see CGraphicsContext::SetWordJustification()
1.2825 +*/
1.2826 +EXPORT_C void CDirectGdiContext::SetWordJustification(TInt aExcessWidth, TInt aNumGaps)
1.2827 + {
1.2828 + GRAPHICS_TRACE2("CDirectGdiContext::SetWordJustification(%d,%d)", aExcessWidth, aNumGaps);
1.2829 + if (aExcessWidth <= 0 || aNumGaps <= 0)
1.2830 + {
1.2831 + iWordJustExcess = 0;
1.2832 + iWordJustNum = 0;
1.2833 + }
1.2834 + else
1.2835 + {
1.2836 + iWordJustExcess = aExcessWidth;
1.2837 + iWordJustNum = aNumGaps;
1.2838 + }
1.2839 + }
1.2840 +
1.2841 +
1.2842 +/**
1.2843 +Sets the underline style for all subsequently drawn text.
1.2844 +The function provides a concrete implementation of the pure virtual function CGraphicsContext::SetUnderlineStyle().
1.2845 +The function behaviour is the same as documented in that class.
1.2846 +
1.2847 +@param aUnderlineStyle The underline style to be used.
1.2848 +
1.2849 +@see CGraphicsContext::SetUnderlineStyle()
1.2850 +*/
1.2851 +EXPORT_C void CDirectGdiContext::SetUnderlineStyle(DirectGdi::TFontUnderline aUnderlineStyle)
1.2852 + {
1.2853 + GRAPHICS_TRACE1("CDirectGdiContext::SetWordJustification(%d)", aUnderlineStyle);
1.2854 + iUnderline = aUnderlineStyle;
1.2855 + }
1.2856 +
1.2857 +
1.2858 +/**
1.2859 +Sets the strikethrough style for all subsequently drawn text.
1.2860 +The function provides a concrete implementation of the pure virtual function CGraphicsContext::SetStrikethroughStyle().
1.2861 +The function behaviour is the same as documented in that class.
1.2862 +
1.2863 +@param aStrikethroughStyle The strikethrough style to be used.
1.2864 +
1.2865 +@see CGraphicsContext::SetStrikethroughStyle()
1.2866 +*/
1.2867 +EXPORT_C void CDirectGdiContext::SetStrikethroughStyle(DirectGdi::TFontStrikethrough aStrikethroughStyle)
1.2868 + {
1.2869 + GRAPHICS_TRACE1("CDirectGdiContext::SetStrikethroughStyle(%d)", aStrikethroughStyle);
1.2870 + iStrikethrough = aStrikethroughStyle;
1.2871 + }
1.2872 +
1.2873 +
1.2874 +/**
1.2875 +Sets the bitmap to be used as the brush pattern when EPatternedBrush is selected.
1.2876 +The DirectGDI generic layer owns the bitmap and will keep the bitmap until ResetBrushPattern() is called.
1.2877 +
1.2878 +The client may modify the content of the bitmap used as the brush pattern. If this is done after
1.2879 +issuing drawing commands there is no guarantee which bitmap content will be used as brush pattern.
1.2880 +Clients must call Finish() on the driver before modifying the bitmap content if they want a guaranteed
1.2881 +result that the previously issued drawing commands will be drawn using the old bitmap brush pattern.
1.2882 +
1.2883 +In the event of a failure, the error state is set to KErrCouldNotConnect if no connection to the font
1.2884 +and bitmap server could be made, KErrBadHandle if the handle of the bitmap is null, KErrUnknown if
1.2885 +no bitmap could be found with the specified handle number, otherwise one of the system-wide error codes.
1.2886 +
1.2887 +@see CDirectGdiContext::SetBrushStyle()
1.2888 +@see CDirectGdiContext::SetBrushOrigin()
1.2889 +@see CDirectGdiContext::ResetBrushPattern()
1.2890 +
1.2891 +@param aBitmap Bitmap that will be used as the brush pattern.
1.2892 +
1.2893 +@pre Bitmap is fully constructed.
1.2894 +@post Bitmap will be used as the brush pattern for subsequent drawing operations when EPatternedBrush
1.2895 + is selected. It remains in effect until ResetBrushPattern() is called.
1.2896 +*/
1.2897 +EXPORT_C void CDirectGdiContext::SetBrushPattern(const CFbsBitmap& aBitmap)
1.2898 + {
1.2899 + GRAPHICS_TRACE1("CDirectGdiContext::SetBrushPattern(%d)", aBitmap.Handle());
1.2900 + SetBrushPattern(aBitmap.Handle());
1.2901 + }
1.2902 +
1.2903 +
1.2904 +/**
1.2905 +Sets the bitmap to be used as the brush pattern when EPatternedBrush is selected.
1.2906 +The DirectGDI generic layer owns the bitmap and will keep the bitmap until ResetBrushPattern() is called.
1.2907 +If the client modifies the content of the bitmap used as the brush pattern after issuing any drawing
1.2908 +commands that uses that brush pattern, the method does not guarantee whether the old bitmap
1.2909 +content or the new one will be used as brush pattern. Clients must call Finish() on the driver
1.2910 +before modifying the bitmap content if they want a guaranteed result that the previously issued
1.2911 +drawing commands will be drawn using the old bitmap brush pattern.
1.2912 +
1.2913 +In the event of a failure, the error state is set to KErrCouldNotConnect if no connection to the font
1.2914 +and bitmap server could be made, KErrBadHandle if the handle is null, KErrUnknown if no bitmap could
1.2915 +be found with the specified handle number, otherwise one of the system-wide error codes.
1.2916 +
1.2917 +@param aFbsBitmapHandle Bitmap handle that will be used as the brush pattern.
1.2918 +
1.2919 +@pre Bitmap belonging to the handle is fully constructed.
1.2920 +@post Bitmap will be used as the brush pattern for subsequent drawing operations when EPatternedBrush
1.2921 + is selected. It remains in effect until ResetBrushPattern() is called.
1.2922 +@panic DGDI 8, if aFbsBitmapHandle is 0.
1.2923 +*/
1.2924 +EXPORT_C void CDirectGdiContext::SetBrushPattern(TInt aFbsBitmapHandle)
1.2925 + {
1.2926 + GRAPHICS_TRACE1("CDirectGdiContext::SetBrushPattern(%d)", aFbsBitmapHandle);
1.2927 + if (aFbsBitmapHandle == KNullHandle)
1.2928 + {
1.2929 + iDriver.SetError(KErrBadHandle);
1.2930 + return;
1.2931 + }
1.2932 +
1.2933 + // Check we're not already using the passed brush pattern
1.2934 + if (iBrushPattern.Handle() == aFbsBitmapHandle)
1.2935 + {
1.2936 + return;
1.2937 + }
1.2938 +
1.2939 + // Delete any previously saved brush pattern
1.2940 + CleanUpBrushPattern();
1.2941 +
1.2942 + TInt result = iBrushPattern.Duplicate(aFbsBitmapHandle);
1.2943 + if (result == KErrNone)
1.2944 + {
1.2945 + result = iEngine->SetBrushPattern(iBrushPattern);
1.2946 + }
1.2947 +
1.2948 + if (result == KErrNone)
1.2949 + {
1.2950 + iBrushPatternUsed = ETrue;
1.2951 + }
1.2952 + else
1.2953 + {
1.2954 + iDriver.SetError(result);
1.2955 + }
1.2956 +
1.2957 + return;
1.2958 + }
1.2959 +
1.2960 +
1.2961 +/**
1.2962 +Selects the font to be used for text drawing.
1.2963 +Notes:
1.2964 +When the font is no longer required, use ResetFont() to free up the memory used.
1.2965 +If SetFont() is used again without using ResetFont() then the previous font is reset
1.2966 +automatically. If no font has been selected, and an attempt is made to draw text with
1.2967 +DrawText(), then a panic occurs.
1.2968 +
1.2969 +@see CDirectGdiContext::ResetFont()
1.2970 +@see CDirectGdiContext::DrawText()
1.2971 +
1.2972 +@param aFont The font to be used.
1.2973 +
1.2974 +@panic DGDI 12, if aFont has an invalid handle or is not a CFbsFont, or the font cannot be duplicated.
1.2975 +*/
1.2976 +EXPORT_C void CDirectGdiContext::SetFont(const CFont* aFont)
1.2977 + {
1.2978 + GRAPHICS_TRACE("CDirectGdiContext::SetFont");
1.2979 + // Note: We pass a ptr in, rather than a reference, because otherwise the caller would almost always have to do complex casting
1.2980 + GRAPHICS_ASSERT_ALWAYS(aFont->TypeUid() == KCFbsFontUid, EDirectGdiPanicInvalidFont);
1.2981 + const CDirectGdiFont* font = reinterpret_cast<const CDirectGdiFont*>(aFont);
1.2982 + GRAPHICS_ASSERT_ALWAYS(font->Handle(), EDirectGdiPanicInvalidFont);
1.2983 +
1.2984 + if (iFont.Handle() == font->Handle())
1.2985 + {
1.2986 + return;
1.2987 + }
1.2988 + ResetFont();
1.2989 + TInt err = iFont.Duplicate(font->Handle());
1.2990 + GRAPHICS_ASSERT_ALWAYS(err == KErrNone, EDirectGdiPanicInvalidFont); // This may seem extreme but it is what BitGdi did
1.2991 + iEngine->SetFont(iFont.Address()->UniqueFontId());
1.2992 + }
1.2993 +
1.2994 +
1.2995 +/**
1.2996 +Copies the content of a rectangular area on the target to another location.
1.2997 +The source rect will be intersected with the target’s full extent.
1.2998 +
1.2999 +@param aOffset Offset from the top left corner of the rectangle to be copied to the top left corner of the copy.
1.3000 +@param aRect Area to be copied.
1.3001 +
1.3002 +@pre The rendering target has been activated.
1.3003 +@post Request to copy an area has been accepted. There is no guarantee that the
1.3004 + request has been processed when this method returns.
1.3005 +
1.3006 +@panic DGDI 7, if the rendering context has not been activated.
1.3007 +*/
1.3008 +EXPORT_C void CDirectGdiContext::CopyRect(const TPoint& aOffset, const TRect& aRect)
1.3009 + {
1.3010 + GRAPHICS_TRACE("CDirectGdiContext::CopyRect");
1.3011 + if (aRect.IsEmpty() || aOffset == TPoint(0,0))
1.3012 + {
1.3013 + return;
1.3014 + }
1.3015 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3016 + iEngine->CopyRect(aOffset, aRect);
1.3017 + }
1.3018 +
1.3019 +/**
1.3020 +Copies all settings from the specified DirectGDI context.
1.3021 +
1.3022 +@param aGc The DirectGDI context whose settings are to be copied.
1.3023 +*/
1.3024 +EXPORT_C void CDirectGdiContext::CopySettings(const CDirectGdiContext& aGc)
1.3025 + {
1.3026 + GRAPHICS_TRACE("CDirectGdiContext::CopySettings");
1.3027 + SetOrigin(aGc.iOrigin);
1.3028 + SetFont(&(aGc.iFont));
1.3029 + SetCharJustification(aGc.iCharJustExcess, aGc.iCharJustNum);
1.3030 + SetWordJustification(aGc.iWordJustExcess, aGc.iWordJustNum);
1.3031 + iLastPrintPosition = aGc.iLastPrintPosition;
1.3032 + SetStrikethroughStyle(aGc.iStrikethrough);
1.3033 + SetUnderlineStyle(aGc.iUnderline);
1.3034 + SetPenColor(aGc.iPenColor);
1.3035 + SetPenSize(aGc.iPenSize);
1.3036 + SetPenStyle(aGc.iPenStyle);
1.3037 + SetDrawMode(aGc.iDrawMode);
1.3038 + SetTextShadowColor(aGc.iTextShadowColor);
1.3039 + SetBrushColor(aGc.iBrushColor);
1.3040 + SetBrushStyle(aGc.iBrushStyle);
1.3041 + if(aGc.iBrushPattern.Handle())
1.3042 + {
1.3043 + SetBrushPattern(aGc.iBrushPattern.Handle());
1.3044 + }
1.3045 + iBrushPatternUsed = aGc.iBrushPatternUsed;
1.3046 + SetBrushOrigin(aGc.iBrushOrigin);
1.3047 + }
1.3048 +
1.3049 +/**
1.3050 +Updates the justification settings.
1.3051 +This function assumes that NoJustifyAutoUpdate() has not been used.
1.3052 +
1.3053 +@param aText The text for which justification is to be adjusted.
1.3054 +@param aParam Parameters used in drawing text.
1.3055 +
1.3056 +@panic DGDI 7, if the rendering context has not been activated.
1.3057 +@panic DGDI 13, if NoJustifyAutoUpdate() had been called prior to this.
1.3058 +*/
1.3059 +EXPORT_C void CDirectGdiContext::UpdateJustification(const TDesC& aText, const DirectGdi::TTextParameters* aParam)
1.3060 + {
1.3061 + GRAPHICS_TRACE("CDirectGdiContext::UpdateJustification");
1.3062 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3063 + GRAPHICS_ASSERT_ALWAYS(iAutoUpdateJustification, EDirectGdiPanicAutoUpdateJustificationUsed);
1.3064 + if (((iCharJustNum < 1) || (iCharJustExcess == 0)) && ((iWordJustNum < 1) || (iWordJustExcess < 1)))
1.3065 + {
1.3066 + return;
1.3067 + }
1.3068 +
1.3069 + TInt length = aText.Length();
1.3070 + CFont::TPositionParam param;
1.3071 + param.iText.Set(aText); // Set the start of the string
1.3072 + if (aParam)
1.3073 + {
1.3074 + length = aParam->iEnd;
1.3075 + param.iPosInText = aParam->iStart;
1.3076 + }
1.3077 + TInt excess = 0;
1.3078 + TInt glyphs = 0;
1.3079 + RShapeInfo shapeInfo;
1.3080 + for (TInt count = 0; count < length; count++)
1.3081 + {
1.3082 + if ((iCharJustNum > 0) && (iCharJustExcess != 0))
1.3083 + {
1.3084 + excess += CGraphicsContext::JustificationInPixels(iCharJustExcess, iCharJustNum);
1.3085 + }
1.3086 + if ((iWordJustNum > 0) && (iWordJustExcess > 0) && (aText[count] == ' '))
1.3087 + {
1.3088 + excess += CGraphicsContext::JustificationInPixels(iWordJustExcess, iWordJustNum);
1.3089 + }
1.3090 + if (iCharJustNum < (glyphs + length - count)) // there's at least 1 combined glyph to come
1.3091 + {
1.3092 + // otherwise we can skip this slow bit and just increment
1.3093 + if (iFont.GetCharacterPosition2(param, shapeInfo))
1.3094 + {
1.3095 + count = param.iPosInText - 1; // -1 'cos it gets incremented anyway
1.3096 + }
1.3097 + }
1.3098 + glyphs++;
1.3099 + }
1.3100 + if (shapeInfo.IsOpen())
1.3101 + {
1.3102 + shapeInfo.Close();
1.3103 + }
1.3104 + iLastPrintPosition.iX += excess;
1.3105 + }
1.3106 +
1.3107 +
1.3108 +/**
1.3109 +Updates the justification for vertical text.
1.3110 +This function assumes that NoJustifyAutoUpdate() has not been used.
1.3111 +
1.3112 +@param aText The text for which justification is to be adjusted.
1.3113 +@param aParam Parameters used in drawing text.
1.3114 +@param aUp ETrue, if text is to be justified upwards; EFalse, if text is to be justified downwards.
1.3115 +
1.3116 +@panic DGDI 7, if the rendering context has not been activated.
1.3117 +@panic DGDI 13, if NoJustifyAutoUpdate() had been called prior to this.
1.3118 +*/
1.3119 +EXPORT_C void CDirectGdiContext::UpdateJustificationVertical(const TDesC& aText, const DirectGdi::TTextParameters* aParam, TBool aUp)
1.3120 + {
1.3121 + GRAPHICS_TRACE("CDirectGdiContext::UpdateJustificationVertical");
1.3122 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3123 + GRAPHICS_ASSERT_ALWAYS(iAutoUpdateJustification, EDirectGdiPanicAutoUpdateJustificationUsed);
1.3124 +
1.3125 + if (((iCharJustNum < 1) || (iCharJustExcess == 0)) && ((iWordJustNum < 1) || (iWordJustExcess < 1)))
1.3126 + {
1.3127 + return;
1.3128 + }
1.3129 +
1.3130 + TInt length = aText.Length();
1.3131 + CFont::TPositionParam param;
1.3132 + param.iText.Set(aText); // Set the start of the string
1.3133 + if (aParam)
1.3134 + {
1.3135 + length = aParam->iEnd;
1.3136 + param.iPosInText = aParam->iStart;
1.3137 + }
1.3138 + TInt excess = 0;
1.3139 + TInt glyphs = 0;
1.3140 + RShapeInfo shapeInfo;
1.3141 + for (TInt count = 0; count < length; count++)
1.3142 + {
1.3143 + if ((iCharJustNum > 0) && (iCharJustExcess != 0))
1.3144 + {
1.3145 + excess += CGraphicsContext::JustificationInPixels(iCharJustExcess, iCharJustNum);
1.3146 + }
1.3147 + if ((iWordJustNum > 0) && (iWordJustExcess > 0) && (aText[count] == ' '))
1.3148 + {
1.3149 + excess += CGraphicsContext::JustificationInPixels(iWordJustExcess, iWordJustNum);
1.3150 + }
1.3151 + if (iCharJustNum < (glyphs + length - count)) // there's at least 1 combined glyph to come
1.3152 + {
1.3153 + // otherwise we can skip this slow bit and just increment
1.3154 + if (iFont.GetCharacterPosition2(param, shapeInfo))
1.3155 + {
1.3156 + count = param.iPosInText - 1; // -1 because it gets incremented anyway
1.3157 + }
1.3158 + }
1.3159 + glyphs++;
1.3160 + }
1.3161 + if (shapeInfo.IsOpen())
1.3162 + {
1.3163 + shapeInfo.Close();
1.3164 + }
1.3165 +
1.3166 + if (aUp)
1.3167 + {
1.3168 + iLastPrintPosition.iY -= excess;
1.3169 + }
1.3170 + else
1.3171 + {
1.3172 + iLastPrintPosition.iY += excess;
1.3173 + }
1.3174 + }
1.3175 +
1.3176 +
1.3177 +/**
1.3178 +Selects a font for text drawing but does not take a copy.
1.3179 +The original must not be destroyed until SetFont(), SetFontNoDuplicate(), ResetFont()
1.3180 +or the destructor is called.
1.3181 +
1.3182 +@param aFont A pointer to the font to be used.
1.3183 +@panic DGDI 12, if aFont has no handle or is not a CFbsFont.
1.3184 +*/
1.3185 +EXPORT_C void CDirectGdiContext::SetFontNoDuplicate(const CDirectGdiFont* aFont)
1.3186 + {
1.3187 + GRAPHICS_TRACE("CDirectGdiContext::SetFontNoDuplicate");
1.3188 + // Note: We pass a ptr in, rather than a reference, because otherwise the caller would almost always have to do complex casting
1.3189 + GRAPHICS_ASSERT_ALWAYS(aFont->TypeUid() == KCFbsFontUid, EDirectGdiPanicInvalidFont);
1.3190 + GRAPHICS_ASSERT_ALWAYS(aFont->Handle(), EDirectGdiPanicInvalidFont);
1.3191 +
1.3192 + if (iFont.Handle() == aFont->Handle())
1.3193 + {
1.3194 + return;
1.3195 + }
1.3196 +
1.3197 + ResetFont();
1.3198 + iFont = *aFont;
1.3199 + iEngine->SetFont(iFont.Address()->UniqueFontId());
1.3200 + }
1.3201 +
1.3202 +
1.3203 +/**
1.3204 +Checks to see if a brush pattern is currently set.
1.3205 +
1.3206 +@return ETrue is a brush pattern is currently set, EFalse if no brush pattern is currently set.
1.3207 +*/
1.3208 +EXPORT_C TBool CDirectGdiContext::HasBrushPattern() const
1.3209 + {
1.3210 + GRAPHICS_TRACE("CDirectGdiContext::HasBrushPattern");
1.3211 + return iBrushPatternUsed;
1.3212 + }
1.3213 +
1.3214 +
1.3215 +/**
1.3216 +Tests whether a font is used.
1.3217 +
1.3218 +@return ETrue, if a font is being used; EFalse, otherwise.
1.3219 +*/
1.3220 +EXPORT_C TBool CDirectGdiContext::HasFont() const
1.3221 + {
1.3222 + GRAPHICS_TRACE("CDirectGdiContext::HasFont");
1.3223 + TBool result = EFalse;
1.3224 + if (iFont.Handle() != KNullHandle)
1.3225 + result = ETrue;
1.3226 + return result;
1.3227 + }
1.3228 +
1.3229 +
1.3230 +/**
1.3231 +Externalises the context and the drawing engine object to the write stream.
1.3232 +It is important that the font and brush bitmap of the GC is maintained between
1.3233 +calls to ExternalizeL() and InternalizeL(). The font handle and brush bitmap handle
1.3234 +is externalised, not the underlying data. This is done for performance reasons.
1.3235 +
1.3236 +@param aWriteStream Write stream.
1.3237 +
1.3238 +@pre None.
1.3239 +@post The context and drawing engine object states are written to the write stream.
1.3240 +
1.3241 +@see MDirectGdiEngine::InternalizeL
1.3242 +@leave If there was an error writing to the write stream.
1.3243 +*/
1.3244 +EXPORT_C void CDirectGdiContext::ExternalizeL(RWriteStream& aWriteStream)
1.3245 + {
1.3246 + GRAPHICS_TRACE("CDirectGdiContext::ExternalizeL");
1.3247 + aWriteStream << KDirectGDIContext_VerNo;
1.3248 + iEngine->ExternalizeL(aWriteStream);
1.3249 +
1.3250 + aWriteStream << iOrigin;
1.3251 + aWriteStream.WriteInt32L(iFont.Handle());
1.3252 + aWriteStream.WriteInt32L(iCharJustExcess);
1.3253 + aWriteStream.WriteInt32L(iCharJustNum);
1.3254 + aWriteStream.WriteInt32L(iWordJustExcess);
1.3255 + aWriteStream.WriteInt32L(iWordJustNum);
1.3256 + aWriteStream << iLastPrintPosition;
1.3257 + aWriteStream.WriteUint8L(iStrikethrough);
1.3258 + aWriteStream.WriteUint8L(iUnderline);
1.3259 + aWriteStream << iPenColor;
1.3260 + aWriteStream.WriteUint32L(iPenSize.iWidth);
1.3261 + aWriteStream.WriteUint32L(iPenSize.iHeight);
1.3262 + aWriteStream.WriteUint8L(iPenStyle);
1.3263 + aWriteStream.WriteUint8L(iDrawMode);
1.3264 + aWriteStream << iTextShadowColor;
1.3265 + aWriteStream << iBrushColor;
1.3266 + aWriteStream.WriteInt32L(iBrushPattern.Handle());
1.3267 + aWriteStream.WriteUint8L(iBrushStyle);
1.3268 + aWriteStream.WriteUint8L(iBrushPatternUsed);
1.3269 + aWriteStream << iBrushOrigin;
1.3270 + aWriteStream.WriteUint8L(iAutoUpdateJustification);
1.3271 + }
1.3272 +
1.3273 +
1.3274 +/**
1.3275 +Internalises the context and the drawing engine object from the read stream.
1.3276 +It is important that the font and brush bitmap of the GC is maintained between
1.3277 +calls to ExternalizeL() and InternalizeL(). The font handle and brush bitmap handle
1.3278 +is externalised, not the underlying data. This is done for performance reasons.
1.3279 +
1.3280 +@param aReadStream Read stream.
1.3281 +
1.3282 +@pre The font has not been released since the last call of CDirectGdiContext::ExternalizeL on the stream
1.3283 +@pre The handle of the brush pattern bitmap has not been closed since the call to CDirectGdiContext::ExternalizeL on the stream.
1.3284 +@post The context and drawing engine object states are updated with the values from the read stream.
1.3285 +
1.3286 +@see MDirectGdiEngine::ExternalizeL
1.3287 +@leave If there was an error reading from the read stream.
1.3288 +*/
1.3289 +EXPORT_C void CDirectGdiContext::InternalizeL(RReadStream& aReadStream)
1.3290 + {
1.3291 + GRAPHICS_TRACE("CDirectGdiContext::InternalizeL");
1.3292 + TUint16 archiveVerNo = 0;
1.3293 + aReadStream >> archiveVerNo;
1.3294 + iEngine->InternalizeL(aReadStream);
1.3295 +
1.3296 + TPoint origin;
1.3297 + aReadStream >> origin;
1.3298 + SetOrigin(origin);
1.3299 + ResetFont();
1.3300 + TInt fontHandle = aReadStream.ReadInt32L();
1.3301 + if(fontHandle)
1.3302 + {
1.3303 + TInt res = iFont.Duplicate(fontHandle);
1.3304 + if(res == KErrNone)
1.3305 + {
1.3306 + iEngine->SetFont(iFont.Address()->UniqueFontId());
1.3307 + }
1.3308 + else
1.3309 + {
1.3310 + iDriver.SetError(res);
1.3311 + }
1.3312 + }
1.3313 + iCharJustExcess = aReadStream.ReadUint32L();
1.3314 + iCharJustNum = aReadStream.ReadUint32L();
1.3315 + iWordJustExcess = aReadStream.ReadUint32L();
1.3316 + iWordJustNum = aReadStream.ReadUint32L();
1.3317 + aReadStream >> iLastPrintPosition;
1.3318 + iStrikethrough = (DirectGdi::TFontStrikethrough)aReadStream.ReadUint8L();
1.3319 + iUnderline = (DirectGdi::TFontUnderline)aReadStream.ReadUint8L();
1.3320 + TRgb penColor;
1.3321 + aReadStream >> penColor;
1.3322 + SetPenColor(penColor);
1.3323 + TSize penSize;
1.3324 + penSize.iWidth = aReadStream.ReadUint32L();
1.3325 + penSize.iHeight = aReadStream.ReadUint32L();
1.3326 + SetPenSize(penSize);
1.3327 + DirectGdi::TPenStyle penStyle = (DirectGdi::TPenStyle)aReadStream.ReadUint8L();
1.3328 + SetPenStyle(penStyle);
1.3329 + DirectGdi::TDrawMode drawMode = (DirectGdi::TDrawMode)aReadStream.ReadUint8L();
1.3330 + SetDrawMode(drawMode);
1.3331 + TRgb textShadowColor;
1.3332 + aReadStream >> textShadowColor;
1.3333 + SetTextShadowColor(textShadowColor);
1.3334 + TRgb brushColor;
1.3335 + aReadStream >> brushColor;
1.3336 + SetBrushColor(brushColor);
1.3337 + TInt patternHandle = aReadStream.ReadInt32L();
1.3338 + if (patternHandle)
1.3339 + {
1.3340 + // Brush pattern must be set before style, otherwise there'll be a panic!
1.3341 + SetBrushPattern(patternHandle);
1.3342 + }
1.3343 + DirectGdi::TBrushStyle brushStyle;
1.3344 + brushStyle = (DirectGdi::TBrushStyle)aReadStream.ReadInt8L();
1.3345 + SetBrushStyle(brushStyle);
1.3346 + iBrushPatternUsed = (TBool)aReadStream.ReadUint8L();
1.3347 + TPoint brushOrigin;
1.3348 + aReadStream >> brushOrigin;
1.3349 + SetBrushOrigin(brushOrigin);
1.3350 + iAutoUpdateJustification = (TBool)aReadStream.ReadUint8L();
1.3351 + }
1.3352 +
1.3353 +
1.3354 +/**
1.3355 +Retrieves the currently set brush colour.
1.3356 +
1.3357 +@return The current brush colour.
1.3358 +*/
1.3359 +EXPORT_C TRgb CDirectGdiContext::BrushColor() const
1.3360 + {
1.3361 + return iBrushColor;
1.3362 + }
1.3363 +
1.3364 +
1.3365 +/**
1.3366 +Retrieves the currently set pen colour.
1.3367 +
1.3368 +@return The current pen colour.
1.3369 +*/
1.3370 +EXPORT_C TRgb CDirectGdiContext::PenColor() const
1.3371 + {
1.3372 + return iPenColor;
1.3373 + }
1.3374 +
1.3375 +/**
1.3376 +Retrieves the currently set text shadow colour.
1.3377 +
1.3378 +@return The current text shadow colour.
1.3379 +*/
1.3380 +EXPORT_C TRgb CDirectGdiContext::TextShadowColor() const
1.3381 + {
1.3382 + return iTextShadowColor;
1.3383 + }
1.3384 +
1.3385 +/**
1.3386 +Draws an image based resource which may be generated using non-native rendering API such as OpenGL ES
1.3387 +or OpenVG. The resource will be drawn at the specified position in its original size with orientation
1.3388 +according to the specified rotation parameter. The current clipping region applies. The resource can be
1.3389 +drawn rotated using the DirectGdi::TGraphicsRotation enum which defines possible rotation values in
1.3390 +clockwise degrees.
1.3391 +
1.3392 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.3393 +
1.3394 +@param aPos The position of the top-left corner of the resource.
1.3395 +@param aSource The resource to be drawn.
1.3396 +@param aRotation The rotation to be applied to the resource before it is drawn. The default value is DirectGdi::EGraphicsRotationNone.
1.3397 +
1.3398 +@pre Drawing context has been activated on a rendering target. The resource has been fully constructed.
1.3399 +@post Request to draw resource has been accepted. There is no guarantee that the request has been completed
1.3400 + when this method returns.
1.3401 +
1.3402 +@panic DGDI 7, if the rendering context has not been activated.
1.3403 +*/
1.3404 +EXPORT_C void CDirectGdiContext::DrawResource(
1.3405 + const TPoint& aPos,
1.3406 + const RDirectGdiDrawableSource& aSource,
1.3407 + DirectGdi::TGraphicsRotation aRotation)
1.3408 + {
1.3409 + GRAPHICS_TRACE("CDirectGdiContext::DrawResource");
1.3410 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3411 +
1.3412 + if (aSource.Handle() != KNullHandle)
1.3413 + {
1.3414 + iEngine->DrawResource(aPos, aSource, aRotation);
1.3415 + }
1.3416 + else
1.3417 + {
1.3418 + iDriver.SetError(KErrBadHandle);
1.3419 + }
1.3420 + }
1.3421 +
1.3422 +/**
1.3423 +Draws an image based resource. The resource will be rendered to the given destination rectangle on
1.3424 +rendering target in its original dimensions with orientation according to the specified rotation parameter.
1.3425 +Drawing will be clipped to the given destination rectangle. The current clipping region applies.
1.3426 +
1.3427 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.3428 +
1.3429 +@param aDestRect Destination rectangle to which the resource will be rendered.
1.3430 +@param aSource The resource to be drawn.
1.3431 +@param aRotation Rotation to be applied to the resource before it is drawn. Default value is DirectGdi::EGraphicsRotationNone.
1.3432 +
1.3433 +@pre Drawing context has been activated on a rendering target. The resource has been fully constructed.
1.3434 +@post Request to draw resource has been accepted. There is no guarantee that the request has been completed
1.3435 + when this method returns.
1.3436 +
1.3437 +@panic DGDI 7, if the rendering context has not been activated.
1.3438 +*/
1.3439 +EXPORT_C void CDirectGdiContext::DrawResource(const TRect& aDestRect,
1.3440 + const RDirectGdiDrawableSource& aSource,
1.3441 + DirectGdi::TGraphicsRotation aRotation)
1.3442 + {
1.3443 + GRAPHICS_TRACE("CDirectGdiContext::DrawResource");
1.3444 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3445 +
1.3446 + if (aSource.Handle() != KNullHandle)
1.3447 + {
1.3448 + if ((aDestRect.Width() > 0) && (aDestRect.Height() > 0))
1.3449 + {
1.3450 + iEngine->DrawResource(aDestRect, aSource, aRotation);
1.3451 + }
1.3452 + }
1.3453 + else
1.3454 + {
1.3455 + iDriver.SetError(KErrBadHandle);
1.3456 + }
1.3457 + }
1.3458 +
1.3459 +
1.3460 +/**
1.3461 +Draws an image based resource. The resource is rendered into the given destination rectangle.
1.3462 +Scaling (stretching or compression) applies if the destination rectangle is different from the
1.3463 +source rectangle. The resource orientation is set based on the specified rotation parameter
1.3464 +before scaling and drawing operations are performed.
1.3465 +
1.3466 +If the user modifies the content of the resource after issuing a DrawResource() command (from the
1.3467 +same thread), the adaptation must make sure that the user’s operations are serialised within
1.3468 +that thread, for example, DrawResource() is processed before the modify operations. The adaptation
1.3469 +does not guarantee the result if the resource modification is performed from threads other than
1.3470 +the one that issued the DrawResource() command. To achieve a guaranteed result in that case, users
1.3471 +must perform synchronisation between any threads that operate on the resource and issue Finish()
1.3472 +on the driver whenever necessary. When using other renderers or mappings, synchronisation is needed
1.3473 +even when this is from within the same thread.
1.3474 +
1.3475 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.3476 +
1.3477 +@param aDestRect The destination rectangle to which the resource will be rendered.
1.3478 +@param aSource The resource to draw.
1.3479 +@param aSrcRect The source rectangle specifying the area/sub-area of the resource to be rendered.
1.3480 +@param aRotation Rotation to be applied to the resource before it is drawn
1.3481 +
1.3482 +@pre The rendering target has been activated. The resource has been fully constructed.
1.3483 +@post Request to draw an image based resource has been accepted. There is no guarantee that the
1.3484 + request has been completed when this method returns.
1.3485 +
1.3486 +@panic DGDI 7, if the rendering context has not been activated.
1.3487 +*/
1.3488 +EXPORT_C void CDirectGdiContext::DrawResource(
1.3489 + const TRect& aDestRect,
1.3490 + const RDirectGdiDrawableSource& aSource,
1.3491 + const TRect& aSrcRect,
1.3492 + DirectGdi::TGraphicsRotation aRotation)
1.3493 + {
1.3494 + GRAPHICS_TRACE("CDirectGdiContext::DrawResource");
1.3495 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3496 +
1.3497 + if (aSource.Handle() != KNullHandle)
1.3498 + {
1.3499 + if ((aDestRect.Width() > 0) && (aDestRect.Height() > 0)
1.3500 + && (aSrcRect.Width() > 0) && (aSrcRect.Height() > 0))
1.3501 + {
1.3502 + iEngine->DrawResource(aDestRect, aSource, aSrcRect, aRotation);
1.3503 + }
1.3504 + }
1.3505 + else
1.3506 + {
1.3507 + iDriver.SetError(KErrBadHandle);
1.3508 + }
1.3509 + }
1.3510 +
1.3511 +
1.3512 +/**
1.3513 +Draws a non-image based resource. The resource will be rendered into the given destination rectangle.
1.3514 +The current clipping region applies. The adaptation is free to interpret the parameters and may define
1.3515 +their own rules on how to handle the rendering of a non-image based resource.
1.3516 +
1.3517 +In the event of a failure, the error state is set to one of the system-wide error codes.
1.3518 +
1.3519 +@param aDestRect The destination rectangle to which the resource will be rendered.
1.3520 +@param aSource The resource.
1.3521 +@param aParam Parameters specifying how to draw the resource.
1.3522 +
1.3523 +@pre The rendering target has been activated. The resource has been fully constructed.
1.3524 +@post Request to draw a non-image based resource has been accepted.
1.3525 + There is no guarantee that the request has been completed when this method returns.
1.3526 +
1.3527 +@panic DGDI 7, if the rendering context has not been activated.
1.3528 +*/
1.3529 +EXPORT_C void CDirectGdiContext::DrawResource(
1.3530 + const TRect& aDestRect,
1.3531 + const RDirectGdiDrawableSource& aSource,
1.3532 + const TDesC8& aParam)
1.3533 + {
1.3534 + GRAPHICS_TRACE("CDirectGdiContext::DrawResource");
1.3535 + GRAPHICS_ASSERT_ALWAYS(iActivated, EDirectGdiPanicContextNotActivated);
1.3536 +
1.3537 + if (aSource.Handle() != KNullHandle)
1.3538 + {
1.3539 + if ((aDestRect.Width() > 0) && (aDestRect.Height() > 0))
1.3540 + {
1.3541 + iEngine->DrawResource(aDestRect, aSource, aParam);
1.3542 + }
1.3543 + }
1.3544 + else
1.3545 + {
1.3546 + iDriver.SetError(KErrBadHandle);
1.3547 + }
1.3548 + }
1.3549 +
1.3550 +/**
1.3551 +Retrieves a pointer to an instance of the appropriate extension interface implementation.
1.3552 +
1.3553 +@param aInterfaceId Interface identifier of the interface to be retrieved.
1.3554 +@param aInterface On return, holds the specified interface, or NULL if the interface cannot be found.
1.3555 +
1.3556 +@pre None.
1.3557 +@post None.
1.3558 +
1.3559 +@return KErrNone If the interface is supported, KErrNotSupported otherwise.
1.3560 + */
1.3561 +EXPORT_C TInt CDirectGdiContext::GetInterface(TUid aInterfaceId, TAny*& aInterface)
1.3562 + {
1.3563 + GRAPHICS_TRACE("CDirectGdiContext::GetInterface");
1.3564 + return iEngine->GetInterface(aInterfaceId, aInterface);
1.3565 + }
1.3566 +
1.3567 +/**
1.3568 +Release the brush pattern's handle, and mark it as no longer used.
1.3569 +*/
1.3570 +void CDirectGdiContext::CleanUpBrushPattern()
1.3571 + {
1.3572 + iBrushPattern.Reset();
1.3573 + iBrushPatternUsed = EFalse;
1.3574 + }
1.3575 +
1.3576 +/**
1.3577 +@internalTechnology
1.3578 +
1.3579 +Returns the baseline correction associated with this font.
1.3580 +This value is used to alter the underline/strikethrough position applied to linked fonts.
1.3581 +
1.3582 +@return The baseline correction value set by the rasterizer; or 0 if not set
1.3583 +*/
1.3584 +TInt CDirectGdiContext::BaselineCorrection()
1.3585 + {
1.3586 + TOpenFontMetrics metrics;
1.3587 + if (iFont.GetFontMetrics(metrics))
1.3588 + return metrics.BaselineCorrection();
1.3589 + else
1.3590 + return 0;
1.3591 + }