os/graphics/windowing/windowserver/nga/CLIENT/RGC.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 1994-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Shells for window server graphics class
    15 // 
    16 //
    17 
    18 #include <e32std.h>
    19 #include <graphics/wsdrawresource.h>
    20 #include "../SERVER/w32cmd.h"
    21 #include "CLIENT.H"
    22 #include "w32comm.h"
    23 #include "graphicsresourcewrapper.h"
    24 #include <graphics/gdi/gdiconsts.h>
    25 #include <graphics/gdi/gdistructs.h>
    26 #include "graphics/windowserverconstants.h"
    27 
    28 #define KDefaultShadowColor KRgbGray
    29 enum {EPolygonMaxHeaderSize=sizeof(TWsCmdHeader)+sizeof(TWsGcCmdSegmentedDrawPolygonData)};
    30 
    31 //
    32 // class CWindowGc::CPimpl
    33 //
    34 
    35 NONSHARABLE_STRUCT(CWindowGc::CPimpl): public CBase, public MWsDrawResource
    36 /** @internalComponent @released */
    37 	{
    38 public:
    39 	enum TStateType
    40 	    {
    41 	    EStateBrushColor           = 1<<0,
    42 	    EStateBrushStyle           = 1<<1,
    43 	    EStatePenColor             = 1<<2,
    44 	    EStatePenStyle             = 1<<3,
    45 	    EStatePenSize              = 1<<4,           
    46 	    EStateDrawMode             = 1<<5,
    47 	    EStateClippingRect         = 1<<6,
    48         EStateUnderlineStyle       = 1<<7,
    49         EStateStrikethroughStyle   = 1<<8,
    50         EStateWordJustification    = 1<<9,
    51         EStateCharJustification    = 1<<10,
    52 	    };
    53 
    54 	CPimpl(CWindowGc& aGc);
    55 	void WriteAnyPendingStateChanges();
    56 	void ResetPendingState();
    57 	void StorePendingStateChange(TStateType aState, const TAny* aValue);
    58 	TUint32 GetState() const {return iPendingState;}
    59 	void CancelPendingClippingRect();
    60 	~CPimpl();
    61 public: //from MWsDrawResource
    62 	void DrawResource(const TPoint& aPos, const RWsDrawableSource& aSource, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
    63 		{
    64 		iGc.DrawResource(aPos, aSource, aRotation);
    65 		}
    66 	void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
    67 		{
    68 		iGc.DrawResource(aDestRect, aSource, aRotation);
    69 		}
    70 	void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TRect& aSrcRect, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
    71 		{
    72 		iGc.DrawResource(aDestRect, aSource, aSrcRect, aRotation);
    73 		}
    74 	void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TDesC8& aParam)
    75 		{
    76 		iGc.DrawResource(aDestRect, aSource, aParam);
    77 		}
    78 public:
    79 	CWindowGc& iGc;
    80 	CFbsFont* iFont;
    81 	TRgb iShadowColor;
    82     TBool iForceWrite;
    83     TBool iClippingRectSet;
    84         
    85 private:
    86     TUint32 iPendingState;
    87 
    88     // Pending state - used to store requested state changes not yet written to wserv
    89     TRgb iPendingBrushColor;
    90 	TBrushStyle iPendingBrushStyle;
    91     TRgb iPendingPenColor;
    92     TPenStyle iPendingPenStyle;
    93     TSize iPendingPenSize;
    94     TDrawMode iPendingDrawMode;
    95     TRect iPendingClippingRect;
    96     TFontUnderline iPendingUnderlineStyle;
    97     TFontStrikethrough iPendingStrikethroughStyle;
    98     TWsGcCmdSetJustification iPendingWordJustification;
    99     TWsGcCmdSetJustification iPendingCharJustification;
   100 
   101     // Current state - values that have actually been written to wserv
   102     TRgb iCurrentBrushColor;
   103 	TBrushStyle iCurrentBrushStyle;
   104     TRgb iCurrentPenColor;
   105     TPenStyle iCurrentPenStyle;
   106     TSize iCurrentPenSize;
   107     TDrawMode iCurrentDrawMode;
   108     TFontUnderline iCurrentUnderlineStyle;
   109     TFontStrikethrough iCurrentStrikethroughStyle;
   110     TWsGcCmdSetJustification iCurrentWordJustification;
   111     TWsGcCmdSetJustification iCurrentCharJustification;
   112 	};
   113 
   114 CWindowGc::CPimpl::CPimpl(CWindowGc& aGc)
   115 	: iGc(aGc), iFont(NULL), iShadowColor(KDefaultShadowColor)
   116 	{
   117 	ResetPendingState();
   118 	}
   119 
   120 CWindowGc::CPimpl::~CPimpl()
   121 	{
   122 	iFont = NULL;
   123 	}
   124 
   125 void CWindowGc::CPimpl::CancelPendingClippingRect()
   126     {
   127     iPendingState &= ~EStateClippingRect;
   128     }
   129 
   130 void CWindowGc::CPimpl::WriteAnyPendingStateChanges()
   131     {
   132     if (iPendingState == 0)
   133         return;
   134     
   135     if (iPendingState & EStateDrawMode)
   136         {
   137         if (iPendingDrawMode != iCurrentDrawMode)
   138             {
   139             iGc.WriteInt(iPendingDrawMode,EWsGcOpSetDrawMode);
   140             iCurrentDrawMode = iPendingDrawMode;
   141             }
   142         }
   143     
   144     if (iPendingState & EStateBrushStyle)
   145         {
   146         if (iPendingBrushStyle != iCurrentBrushStyle)
   147             {
   148             iGc.WriteInt(iPendingBrushStyle,EWsGcOpSetBrushStyle);
   149             iCurrentBrushStyle = iPendingBrushStyle;
   150             }
   151         }
   152 
   153     if (iPendingState & EStateBrushColor)
   154         {
   155         // Brush colour optimisation more conservative (than for other state changes) because server-side code modifies it without client's knowledge
   156         if (iPendingBrushColor != iCurrentBrushColor || iForceWrite)
   157             {
   158             iGc.WriteInt(iPendingBrushColor.Internal(),EWsGcOpSetBrushColor);
   159             iCurrentBrushColor = iPendingBrushColor;
   160             iForceWrite = EFalse;
   161             }
   162         }
   163 
   164     if (iPendingState & EStatePenStyle)
   165         {
   166         if (iPendingPenStyle != iCurrentPenStyle)
   167             {
   168             iGc.WriteInt(iPendingPenStyle,EWsGcOpSetPenStyle);
   169             iCurrentPenStyle = iPendingPenStyle;   
   170             }
   171         }
   172         
   173     if (iPendingState & EStatePenColor)
   174         {
   175         if (iPendingPenColor != iCurrentPenColor)
   176             {
   177             iGc.WriteInt(iPendingPenColor.Internal(),EWsGcOpSetPenColor);
   178             iCurrentPenColor = iPendingPenColor;            
   179             }
   180         }
   181     
   182     if (iPendingState & EStateClippingRect)
   183         {
   184         iGc.WriteRect(iPendingClippingRect,EWsGcOpSetClippingRect);
   185         iClippingRectSet = ETrue;
   186         }
   187 
   188     if (iPendingState & EStateUnderlineStyle)
   189         {
   190         if (iPendingUnderlineStyle != iCurrentUnderlineStyle)
   191             {
   192             iGc.WriteInt(iPendingUnderlineStyle,EWsGcOpSetUnderlineStyle);
   193             iCurrentUnderlineStyle = iPendingUnderlineStyle;            
   194             }
   195         }
   196     
   197     if (iPendingState & EStateStrikethroughStyle)
   198         {
   199         if (iPendingStrikethroughStyle != iCurrentStrikethroughStyle)
   200             {
   201             iGc.WriteInt(iPendingStrikethroughStyle,EWsGcOpSetStrikethroughStyle);
   202             iCurrentStrikethroughStyle = iPendingStrikethroughStyle;
   203             }
   204         }
   205     
   206     if (iPendingState & EStatePenSize)
   207         {
   208         if (iPendingPenSize != iCurrentPenSize)
   209             {
   210             iGc.WriteSize(iPendingPenSize,EWsGcOpSetPenSize);
   211             iCurrentPenSize = iPendingPenSize;
   212             }
   213         }    
   214 
   215     if (iPendingState & EStateWordJustification)
   216         {
   217         if (iPendingWordJustification.excessWidth != iCurrentWordJustification.excessWidth
   218             || iPendingWordJustification.numGaps != iCurrentWordJustification.numGaps)
   219             {
   220             iGc.Write(&iPendingWordJustification,sizeof(iPendingWordJustification),EWsGcOpSetWordJustification);
   221             iCurrentWordJustification = iPendingWordJustification;
   222             }
   223         }    
   224 
   225     if (iPendingState & EStateCharJustification)
   226         {
   227         if (iPendingCharJustification.excessWidth != iCurrentCharJustification.excessWidth
   228             || iPendingCharJustification.numGaps != iCurrentCharJustification.numGaps)
   229             {
   230             iGc.Write(&iPendingCharJustification,sizeof(iPendingCharJustification),EWsGcOpSetCharJustification);
   231             iCurrentCharJustification = iPendingCharJustification;
   232             }
   233         }    
   234     
   235     iPendingState = 0;
   236     }
   237 
   238 void CWindowGc::CPimpl::ResetPendingState()
   239     {
   240     // This function should only be called from CWindowGc::Reset()
   241     iForceWrite = EFalse;
   242     iPendingState = 0;
   243 
   244     iClippingRectSet = EFalse;
   245 
   246     // The following default values are the same as those used by CFbsBitGc::Reset()
   247     iPendingBrushColor = KRgbWhite;
   248     iPendingBrushStyle = CGraphicsContext::ENullBrush;
   249     iPendingPenColor = KRgbBlack;
   250     iPendingPenStyle = CGraphicsContext::ESolidPen;
   251     iPendingPenSize = TSize(1,1);
   252     iPendingDrawMode = CGraphicsContext::EDrawModePEN;
   253     iPendingUnderlineStyle = EUnderlineOff;    
   254     iPendingStrikethroughStyle = EStrikethroughOff;
   255     iPendingWordJustification.excessWidth = 0;
   256     iPendingWordJustification.numGaps = 0;
   257     iPendingCharJustification.excessWidth = 0;
   258     iPendingCharJustification.numGaps = 0;
   259         
   260     iCurrentBrushColor = iPendingBrushColor;
   261     iCurrentBrushStyle = iPendingBrushStyle;
   262     iCurrentPenColor = iPendingPenColor;
   263     iCurrentPenStyle = iPendingPenStyle;
   264     iCurrentPenSize = iPendingPenSize;    
   265     iCurrentDrawMode = iPendingDrawMode;
   266     iCurrentUnderlineStyle = iPendingUnderlineStyle;
   267     iCurrentStrikethroughStyle = iPendingStrikethroughStyle;
   268     iCurrentWordJustification.excessWidth = iPendingWordJustification.excessWidth;
   269     iCurrentWordJustification.numGaps = iPendingWordJustification.numGaps;
   270     iCurrentCharJustification.excessWidth = iPendingCharJustification.excessWidth;
   271     iCurrentCharJustification.numGaps = iPendingCharJustification.numGaps;
   272     }
   273     
   274 void CWindowGc::CPimpl::StorePendingStateChange(TStateType aState, const TAny* aValue)
   275     {
   276     switch (aState)
   277         {
   278         case EStateBrushColor:
   279             iPendingState |= EStateBrushColor;
   280             iPendingBrushColor = *((TRgb*)(aValue));
   281             break;
   282         case EStateBrushStyle:
   283             iPendingState|=EStateBrushStyle;
   284             iPendingBrushStyle = *((TBrushStyle*)(aValue));
   285             break;
   286         case EStatePenColor:
   287             iPendingState|=EStatePenColor;
   288             iPendingPenColor = *((TRgb*)(aValue));
   289             break;
   290         case EStatePenStyle:
   291             iPendingState|=EStatePenStyle;
   292             iPendingPenStyle = *((TPenStyle*)(aValue));
   293             break;
   294         case EStateDrawMode:
   295             iPendingState|=EStateDrawMode;
   296             iPendingDrawMode = *((TDrawMode*)(aValue));
   297             break;
   298         case EStateClippingRect:
   299             iPendingState|=EStateClippingRect;
   300             iPendingClippingRect = *((TRect*)(aValue));
   301             break;
   302         case EStateUnderlineStyle:
   303             iPendingState|=EStateUnderlineStyle;
   304             iPendingUnderlineStyle = *((TFontUnderline*)(aValue));
   305             break;
   306         case EStateStrikethroughStyle:
   307             iPendingState|=EStateStrikethroughStyle;
   308             iPendingStrikethroughStyle= *((TFontStrikethrough*)(aValue));
   309             break;
   310         case EStatePenSize:
   311             iPendingState|=EStatePenSize;
   312             iPendingPenSize = *((TSize*)(aValue));
   313             break;
   314         case EStateWordJustification:
   315             iPendingState|=EStateWordJustification;
   316             iPendingWordJustification = *((TWsGcCmdSetJustification*)(aValue));
   317             break;
   318         case EStateCharJustification:
   319             iPendingState|=EStateCharJustification;
   320             iPendingCharJustification = *((TWsGcCmdSetJustification*)(aValue));
   321             break;
   322         default:
   323             break;
   324         }
   325     }
   326 
   327 //
   328 // class CWindowGc
   329 //
   330 
   331 EXPORT_C CWindowGc::CWindowGc(CWsScreenDevice *aDevice) : MWsClientClass(aDevice->iBuffer), iPimpl(NULL), iDevice(aDevice)
   332 /** Constructor which creates, but does not initialise a graphics context. 
   333 
   334 @param aDevice Any screen device owned by the same session. Its life time 
   335 should be at least as long as the gc itself. 
   336 @see CWsScreenDevice::CreateContext() */
   337 	{}
   338 
   339 EXPORT_C CWindowGc::~CWindowGc()
   340 /** Destructor. */
   341 	{
   342 	if (iBuffer && iWsHandle)
   343 		Write(EWsGcOpFree);
   344 	delete iPimpl;
   345 	}
   346 
   347 void CWindowGc::WriteTextCommand(TAny *cmd, TInt len,const TDesC &aBuf,TInt opcode,TInt opcodePtr) const
   348 	{
   349 	if ((aBuf.Size()+len)>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)))
   350 		{
   351 		WriteReplyByProvidingRemoteReadAccess(cmd,len,&aBuf,opcodePtr);
   352 		}
   353 	else
   354 		{
   355 		Write(cmd,len,aBuf.Ptr(),aBuf.Size(),opcode);
   356 		}
   357 	}
   358 	
   359 void CWindowGc::WriteTextCommand(TAny *cmd, TInt len,const TDesC8 &aBuf,TInt opcode,TInt opcodePtr) const
   360 	{
   361 	if ((aBuf.Size()+len)>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)))
   362 		{
   363 		WriteReplyByProvidingRemoteReadAccess(cmd,len,&aBuf,opcodePtr);
   364 		}
   365 	else
   366 		{
   367 		Write(cmd,len,aBuf.Ptr(),aBuf.Size(),opcode);
   368 		}
   369 	}	
   370 	
   371 EXPORT_C TInt CWindowGc::Construct()
   372 /** Completes construction. 
   373 
   374 @return KErrNone if successful, otherwise a leave error. 
   375 @panic TW32Panic 17 in debug builds if called on an already constructed object.
   376 This function always causes a flush of the window server buffer. */
   377 	{
   378 	__ASSERT_DEBUG(iWsHandle == KNullHandle, Panic(EW32PanicGraphicDoubleConstruction));
   379 	iPimpl = new CPimpl(*this);
   380 	if (!iPimpl)
   381 		return KErrNoMemory;
   382 	TInt ret;
   383 	if ((ret=iBuffer->WriteReplyWs(EWsClOpCreateGc))<0)
   384 		return(ret);
   385 	iWsHandle=ret;
   386 	return(KErrNone);
   387 	}
   388 
   389 EXPORT_C void CWindowGc::Activate(RDrawableWindow &aDevice)
   390 /** Activates the context for a given window and updates iDevice with the pointer to the screen device of the screen on which aDevice is found.  
   391 
   392 When drawing is complete, the code using the context should call Deactivate(). 
   393 Draw methods invoked after an Activate() will affect the window specified. 
   394 A graphics context can only be active for one window at a time. A panic occurs 
   395 if a draw function is called before calling this function, or if Activate() 
   396 is called twice without an intervening Deactivate().
   397 
   398 @param aWindow The window for which the graphics context is to be activated. */
   399 	{
   400 	TUint devicePointer = WriteReplyInt(aDevice.WsHandle(),EWsGcOpActivate);
   401 	iDevice = (CWsScreenDevice*)devicePointer;
   402 	iPimpl->iForceWrite = ETrue; // needed because brush colour set to window background colour in CWsGc::Activate
   403 	}
   404 
   405 EXPORT_C void CWindowGc::Deactivate()
   406 /** Frees the graphics context to be used with another window. 
   407 
   408 This method should be called when the application has completed drawing to 
   409 the window. */
   410 	{
   411 	iPimpl->ResetPendingState(); // needed because server-side state is reset on deactivation
   412 	Write(EWsGcOpDeactivate);
   413 	iPimpl->iFont=NULL;
   414 	iPimpl->iShadowColor = KDefaultShadowColor;
   415 	}
   416 
   417 //====================Functions from GDI.H===============================
   418 
   419 EXPORT_C CGraphicsDevice* CWindowGc::Device() const
   420 /** Returns a pointer to the device, more specifically a CWsScreenDevice, for the screen that the WindowGc was last activated on.
   421 If the WindowGc has not been activated at all, it then returns the device that was passed to its constructor.
   422 
   423 The user should be careful when calling this function since it can return the screen device of any screen in the system.
   424 Hence, the return value of this function will be useful only if the user is aware of how the WindowGc was used before this function is called.
   425 @return A pointer to the device for the screen that the WindowGc was last activated on or the device passed at construction*/
   426 	{
   427 	return(iDevice);
   428 	}
   429 
   430 EXPORT_C void CWindowGc::SetOrigin(const TPoint &aPoint)
   431 /** Sets the position of the co-ordinate origin. 
   432 
   433 All subsequent drawing operations are then done relative to this origin. The 
   434 default origin is (0,0), the top left corner of the window.
   435 
   436 @param aPoint A point for the origin, default (0,0). 
   437 @see CGraphicsContext::SetOrigin() */
   438 	{
   439 	if ( iPimpl->GetState() & CPimpl::EStateClippingRect )
   440 	    {
   441 	    iPimpl->WriteAnyPendingStateChanges();
   442 	    }
   443 	WritePoint(aPoint,EWsGcOpSetOrigin);
   444 	}
   445 
   446 EXPORT_C void CWindowGc::SetClippingRect(const TRect& aRect)
   447 /** Sets a clipping rectangle.
   448 
   449 Graphics drawn to the window are clipped, so that only items which fall within 
   450 the rectangle are displayed. 
   451 
   452 Note that clipping is additive. If a clipping region has been set using SetClippingRegion() 
   453 then clipping will be to the intersection of that region and this rectangle.
   454 
   455 @param aRect The clipping rectangle in window co-ordinates. Note that this rectangle is 
   456 tranformed by the current drawing co-ordinate origin before it is used. 
   457 The co-ordinate origin is set using SetOrigin().
   458  
   459 @see SetClippingRegion() */
   460 	{
   461     TRect rect(aRect);
   462     iPimpl->StorePendingStateChange(CPimpl::EStateClippingRect, &rect);
   463 	}
   464 
   465 EXPORT_C void CWindowGc::CancelClippingRect()
   466 /** Cancels the clipping rectangle. 
   467 
   468 @see SetClippingRect() */
   469 	{
   470 	if (iPimpl->GetState() & CPimpl::EStateClippingRect)
   471 	    {
   472 	    iPimpl->CancelPendingClippingRect();
   473 	    }
   474 	
   475 	if (iPimpl->iClippingRectSet)
   476 	    {
   477         Write(EWsGcOpCancelClippingRect);
   478 	    }
   479 
   480 	iPimpl->iClippingRectSet = EFalse;
   481 	}
   482 
   483 EXPORT_C TInt CWindowGc::SetClippingRegion(const TRegion &aRegion)
   484 /** Sets the clipping region.
   485 
   486 Drawing is always clipped to the visible area of a window. The region specified 
   487 by this function is in addition to that area.
   488 
   489 This function always causes a flush of the window server buffer.
   490 
   491 @param aRegion The new clipping region in window co-ordinates 
   492 as used in SetClippingRect(). The clipping region is transformed by the current 
   493 drawing origin before use.
   494 
   495 @return KErrNone if successful, KErrNoMemory if there is insufficient memory 
   496 to create the region on the server side, otherwise another error code. 
   497 
   498 @see SetClippingRect() */
   499 	{
   500 	const TInt regionCount=aRegion.Count();
   501 	TPtrC8 ptrRect(reinterpret_cast<const TUint8*>(aRegion.RectangleList()),regionCount*sizeof(TRect));
   502 	return(WriteReplyByProvidingRemoteReadAccess(&regionCount,sizeof(regionCount),&ptrRect,EWsGcOpSetClippingRegion));
   503 	}
   504 
   505 EXPORT_C void CWindowGc::CancelClippingRegion()
   506 /** Cancels the current clipping region. */
   507 	{
   508 	Write(EWsGcOpCancelClippingRegion);
   509 	}
   510 
   511 EXPORT_C void CWindowGc::SetDrawMode(TDrawMode aDrawingMode)
   512 /** Sets the drawing mode. 
   513 
   514 This affects the colour that is actually drawn, because it defines the way 
   515 that the current screen colour logically combines with the current pen colour 
   516 and brush colour. 
   517 
   518 There are 13 drawing modes (see CGraphicsContext::TDrawMode enum), each giving 
   519 different logical combinations of pen, brush and screen colours. Each mode 
   520 is produced by ORing together different combinations of seven drawing mode 
   521 components (see CGraphicsContext::TDrawModeComponents enum).
   522 
   523 The three most important modes are TDrawMode::EDrawModePEN, TDrawMode::EDrawModeNOTSCREEN 
   524 and TDrawMode::EDrawModeXOR. The default drawing mode is TDrawMode::EDrawModePEN.
   525 
   526 The drawing mode is over-ridden for line and shape drawing functions when 
   527 a wide pen line has been selected. It is forced to TDrawMode::EDrawModePEN. 
   528 This is to prevent undesired effects at line joins (vertexes).
   529 
   530 Notes:
   531 
   532 TDrawMode::EDrawModeAND gives a "colour filter" effect. For example:
   533 
   534 ANDing with white gives the original colour 
   535 
   536 ANDing with black gives black 
   537 
   538 TDrawMode::EDrawModeOR gives a "colour boost" effect. For example:
   539 
   540 ORing with black gives the original colour 
   541 
   542 ORing with white gives white 
   543 
   544 TDrawMode::EDrawModeXOR gives an "Exclusive OR" effect. For example:
   545 
   546 white XOR black gives white 
   547 
   548 white XOR white gives black 
   549 
   550 black XOR black gives black
   551 
   552 @param aDrawingMode A drawing mode. 
   553 @see CGraphicsContext::SetDrawMode() */
   554 	{
   555     iPimpl->StorePendingStateChange(CPimpl::EStateDrawMode, &aDrawingMode);
   556 	}
   557 
   558 EXPORT_C void CWindowGc::UseFont(const CFont *aFont)
   559 /** Sets this context's font.
   560 
   561 The font is used for text drawing. If the font is already in the font and 
   562 bitmap server's memory the GDI will share that copy.
   563 
   564 Note that this function must be called prior to drawing text or the calling 
   565 thread will panic.
   566 
   567 @param aFont A device font. 
   568 @see CGraphicsContext::UseFont() */
   569 	{
   570 	if (iPimpl->iFont!=(CFbsFont *)aFont)
   571 		{
   572 		iPimpl->iFont=(CFbsFont *)aFont;
   573 		WriteInt(iPimpl->iFont->Handle(),EWsGcOpUseFont);
   574 		}
   575 	}
   576 
   577 EXPORT_C void CWindowGc::DiscardFont()
   578 /** Discards a font. 
   579 
   580 This frees up the memory used (if the font is not being shared with some other 
   581 process).
   582 
   583 Note that if no font is in use when this function is called, then there is no effect.
   584 
   585 @see CGraphicsContext::DiscardFont() */
   586 	{
   587 	Write(EWsGcOpDiscardFont);
   588 	iPimpl->iFont=NULL;
   589 	}
   590 
   591 EXPORT_C void CWindowGc::SetUnderlineStyle(TFontUnderline aUnderlineStyle)
   592 /** Sets the underline style for all subsequently drawn text.
   593 
   594 @param aUnderlineStyle The underline style: either on or off.
   595 @see CGraphicsContext::SetUnderlineStyle() */
   596 	{
   597     iPimpl->StorePendingStateChange(CPimpl::EStateUnderlineStyle, &aUnderlineStyle);	
   598 	}
   599 
   600 EXPORT_C void CWindowGc::SetStrikethroughStyle(TFontStrikethrough aStrikethroughStyle)
   601 /** Sets the strikethrough style for all subsequently drawn text.
   602 
   603 @param aStrikethroughStyle The strikethrough style: either on or off. 
   604 @see CGraphicsContext::SetStrikethroughStyle() */
   605 	{
   606 	iPimpl->StorePendingStateChange(CPimpl::EStateStrikethroughStyle, &aStrikethroughStyle);
   607 	}
   608 
   609 void CWindowGc::SetJustification(TInt aExcessWidth,TInt aNumGaps, TInt aOpcode)
   610 	{
   611 	TWsGcCmdSetJustification justification;
   612 
   613 	justification.excessWidth=aExcessWidth;
   614 	justification.numGaps=aNumGaps;
   615 
   616 	if (aOpcode == EWsGcOpSetWordJustification)
   617 	    {
   618         iPimpl->StorePendingStateChange(CPimpl::EStateWordJustification, &justification);
   619 	    }
   620 	else if (aOpcode == EWsGcOpSetCharJustification)
   621 	    {
   622         iPimpl->StorePendingStateChange(CPimpl::EStateCharJustification, &justification);
   623 	    }
   624 	}
   625 
   626 EXPORT_C void CWindowGc::SetWordJustification(TInt aExcessWidth,TInt aNumGaps)
   627 /** Sets word justification.
   628 
   629 This function is particularly useful for doing WYSIWYG underlining or strikethrough, 
   630 as it ensures that the lines extend correctly into the gaps between words. It is not 
   631 intended for regular use by developers.
   632 
   633 @param aExcessWidth The excess width (in pixels) to be distributed between 
   634 the specified number of gaps (starting immediately) 
   635 @param aNumGaps The number of gaps between words 
   636 @see CGraphicsContext::SetWordJustification() */
   637 	{
   638 	SetJustification(aExcessWidth, aNumGaps, EWsGcOpSetWordJustification);
   639 	}
   640 
   641 EXPORT_C void CWindowGc::SetCharJustification(TInt aExcessWidth,TInt aNumChars)
   642 /** Sets the character justification.
   643 
   644 This function is used primarily to get accurate WYSIWYG, and is not intended 
   645 for regular use by developers.
   646 
   647 The text line that is to be justified has a certain number of characters (this 
   648 includes the spaces between the words). It also has a distance (in pixels) 
   649 between the end of the last word and the actual end of the line (right hand 
   650 margin, usually). These excess width pixels are distributed amongst all the 
   651 characters, increasing the gaps between them, to achieve full justification 
   652 of the text line.
   653 
   654 This function is particularly useful for WYSIWYG underlining or strikethrough, 
   655 as it ensures that the lines extend into the gaps between characters.
   656 
   657 See CGraphicsContext::SetCharJustification() for more information.
   658 
   659 @param aExcessWidth The excess width (in pixels) to be distributed between 
   660 the specified number of characters. 
   661 @param aNumChars The number of characters involved 
   662 @see CGraphicsContext::SetCharJustification() */
   663 	{
   664 	SetJustification(aExcessWidth, aNumChars, EWsGcOpSetCharJustification);
   665 	}
   666 
   667 EXPORT_C void CWindowGc::SetPenColor(const TRgb &aColor)
   668 /** Sets the pen colour.
   669 
   670 The effective pen colour depends on the drawing mode (see SetDrawMode()).
   671 
   672 The default pen colour is black.
   673 
   674 @param aColor The RGB colour for the pen. 
   675 @see CGraphicsContext::SetPenColor() */
   676 	{
   677 	TRgb color = aColor;
   678     iPimpl->StorePendingStateChange(CPimpl::EStatePenColor, &color);
   679 	}
   680 
   681 EXPORT_C void CWindowGc::SetPenStyle(TPenStyle aPenStyle)
   682 /** Sets the line drawing style for the pen. 
   683 
   684 The pen is used when drawing lines and for the outline of filled shapes. There 
   685 are 6 pen styles (see CGraphicsContext::TPenStyle enum). If no pen style is 
   686 set, the default is TPenStyle::ESolidPen.
   687 
   688 To use a pen style, its full context must be given, e.g. for a null pen: CGraphicsContext::TPenStyle::ENullPen.
   689 
   690 @param aPenStyle A pen style. 
   691 @see CGraphicsContext::SetPenStyle() */
   692 	{
   693 	iPimpl->StorePendingStateChange(CPimpl::EStatePenStyle, &aPenStyle);
   694 	}
   695 
   696 EXPORT_C void CWindowGc::SetPenSize(const TSize& aSize)
   697 /** Sets the line drawing size for the pen.
   698 
   699 Lines of size greater than one pixel are drawn with rounded ends that extend 
   700 beyond the end points, (as if the line is drawn using a circular pen tip of 
   701 the specified size). Rounded ends of lines drawn with a wide pen are always 
   702 drawn in TDrawMode::EDrawModePEN mode, overriding whatever mode has been set 
   703 using SetDrawMode().
   704 
   705 @param aSize A line size, the default being 1 pixel. 
   706 @see CGraphicsContext::SetPenSize() */
   707 	{
   708 	TSize size(aSize);
   709 	iPimpl->StorePendingStateChange(CPimpl::EStatePenSize, &size);
   710 	}
   711 
   712 EXPORT_C void CWindowGc::SetBrushColor(const TRgb &aColor)
   713 /** Sets the brush colour. 
   714 
   715 The effective brush colour depends on the drawing mode (see SetDrawMode()). 
   716 If no brush colour has been set, it defaults to white. However the default 
   717 brush style is null, so when drawing to a window, the default appears to be 
   718 the window's background colour.
   719 
   720 @param aColor The RGB colour for the brush. 
   721 @see CGraphicsContext::SetBrushColor() */
   722 	{
   723 	TRgb color = aColor;
   724     iPimpl->StorePendingStateChange(CPimpl::EStateBrushColor, &color);
   725 	}
   726 
   727 EXPORT_C void CWindowGc::SetBrushStyle(TBrushStyle aBrushStyle)
   728 /** Sets the line drawing style for the brush.
   729 
   730 The GDI provides ten brush styles, including six built-in hatching patterns 
   731 (see CGraphicsContext::TBrushStyle).
   732 
   733 Use TBrushStyle::ENullBrush to draw the outline of a fillable shape on its 
   734 own, without filling.
   735 
   736 If the TBrushStyle::EPatternedBrush style is set, but no bitmap pattern has 
   737 been selected using UseBrushPattern(), then the brush defaults to TBrushStyle::ENullBrush.
   738 
   739 Hatching lines are done in the current brush colour, set using SetBrushColor(). 
   740 Hatching can be overlaid on other graphics. The hatching pattern starts at 
   741 the brush origin, set using SetBrushOrigin().
   742 
   743 @param aBrushStyle The brush style. 
   744 @see CGraphicsContext::SetBrushStyle() */
   745 	{
   746 	iPimpl->StorePendingStateChange(CPimpl::EStateBrushStyle, &aBrushStyle);
   747 	}
   748 EXPORT_C void CWindowGc::SetBrushOrigin(const TPoint &aOrigin)
   749 /** Sets the brush pattern origin. 
   750 
   751 This specifies the position of the pixel in the top left corner of a reference 
   752 pattern tile, (in absolute device co-ordinates). Other copies of the pattern 
   753 tile are then drawn around the reference one. Thus the brush origin can be 
   754 set as the top left corner of a shape.
   755 
   756 The brush pattern may be a built-in style (see SetBrushStyle()), or a bitmap. 
   757 To use a bitmap, the brush must have a pattern set (see UseBrushPattern()) 
   758 and the brush style must be set to TBrushStyle::EPatternedBrush.
   759 
   760 Notes:
   761 
   762 If SetBrushOrigin() is not used, then the origin defaults to (0,0).
   763 
   764 This brush origin remains in effect for all fillable shapes drawn subsequently, 
   765 until a new brush origin is set. Shapes can thus be considered as windows 
   766 onto a continuous pattern field (covering the whole clipping region of a screen 
   767 device, or the whole device area of a printer).
   768 
   769 @param aOrigin The origin point for the brush. 
   770 @see CGraphicsContext::SetBrushOrigin() */
   771 	{
   772 	WritePoint(aOrigin,EWsGcOpSetBrushOrigin);
   773 	}
   774 
   775 EXPORT_C void CWindowGc::UseBrushPattern(const CFbsBitmap *aDevice)
   776 /** Sets the brush pattern to the specified bitmap. 
   777 
   778 For the brush to actually use the bitmap, TBrushStyle::EPatternedBrush must 
   779 be used to set the brush style (see SetBrushStyle()). When the brush pattern 
   780 is no longer required, use DiscardBrushPattern() to free up the memory used, 
   781 (if the bitmap is not being shared). If UseBrushPattern() is used again without 
   782 using DiscardBrushPattern() then the previous pattern is discarded automatically.
   783 
   784 Notes:
   785 
   786 When loading a bitmap, the GDI checks to see if the bitmap is already in memory. 
   787 If the bitmap is already there, then that copy is shared.
   788 
   789 The brush does not need to have a pattern set at all. There are several built-in 
   790 hatching patterns, which can be selected using SetBrushStyle().
   791 
   792 @param aDevice A bitmap pattern for the brush 
   793 @see CGraphicsContext::UseBrushPattern() */
   794 	{
   795 	WriteInt(aDevice->Handle(),EWsGcOpUseBrushPattern);
   796 	AddToBitmapArray(aDevice->Handle());
   797 	}
   798 
   799 EXPORT_C void CWindowGc::DiscardBrushPattern()
   800 /** Discards a non-built-in brush pattern. 
   801 
   802 This frees up the memory used for the bitmap, if it is not being shared by 
   803 another process.
   804 
   805 If no brush pattern has been set when this function is called, it has no effect.
   806 
   807 @see CGraphicsContext::DiscardBrushPattern() */
   808 	{
   809 	Write(EWsGcOpDiscardBrushPattern);
   810 	}
   811 
   812 EXPORT_C void CWindowGc::Plot(const TPoint &aPoint)
   813 /** Draws a single point. 
   814 
   815 The point is drawn with the current pen settings using the current drawing 
   816 mode.
   817 
   818 Note: if the pen size is greater than one pixel, a filled circle of the current 
   819 pen colour is drawn, with the pen size as the diameter and the plotted point 
   820 as the centre. If the pen size is an even number of pixels, the extra pixels 
   821 are drawn below and to the right of the centre. See SetPenSize().
   822 
   823 @param aPoint The point to be drawn. 
   824 @see CGraphicsContext::Plot() */
   825 	{
   826 	iPimpl->WriteAnyPendingStateChanges();
   827 	WritePoint(aPoint,EWsGcOpPlot);
   828 	}
   829 
   830 EXPORT_C void CWindowGc::DrawLine(const TPoint &aPoint1,const TPoint &aPoint2)
   831 /** Draws a straight line between two points.
   832 
   833 @param aPoint1 The point at the start of the line. 
   834 @param aPoint2 The point at the end of the line. 
   835 @see CGraphicsContext::DrawLine() */
   836 	{
   837 	iPimpl->WriteAnyPendingStateChanges();
   838 	TWsGcCmdDrawLine drawLine(aPoint1,aPoint2);
   839 	Write(&drawLine,sizeof(drawLine),EWsGcOpDrawLine);
   840 	}
   841 
   842 EXPORT_C void CWindowGc::MoveTo(const TPoint &aPoint)
   843 /** Moves the internal drawing position relative to the co-ordinate origin, without 
   844 drawing a line.
   845 
   846 A subsequent call to DrawLineTo() or DrawLineBy() will then use the new internal 
   847 drawing position as the start point for the line drawn.
   848 
   849 Notes:
   850 
   851 The operations DrawLine(), DrawLineTo(), DrawLineBy() and DrawPolyline() also 
   852 change the internal drawing position to the last point of the drawn line(s).
   853 
   854 The internal drawing position is set to the co-ordinate origin if no drawing 
   855 or moving operations have yet taken place.
   856 
   857 @param aPoint The point to move the internal drawing position to. 
   858 @see CGraphicsContext::MoveTo()
   859 @see CGraphicsContext::MoveBy() */
   860 	{
   861 	WritePoint(aPoint,EWsGcOpMoveTo);
   862 	}
   863 
   864 EXPORT_C void CWindowGc::MoveBy(const TPoint &aPoint)
   865 /** Moves the internal drawing position by a vector, without drawing a line.
   866 
   867 The internal drawing position is moved relative to its current co-ordinates.
   868 
   869 @param aPoint The vector to move the internal drawing position by. 
   870 @see CGraphicsContext::MoveBy()
   871 @see CGraphicsContext::MoveTo() */
   872 	{
   873 	WritePoint(aPoint,EWsGcOpMoveBy);
   874 	}
   875 
   876 EXPORT_C void CWindowGc::DrawLineTo(const TPoint &aPoint)
   877 /** Draws a straight line from the current internal drawing position to a point.
   878 
   879 @param aPoint The point at the end of the line. 
   880 @see CGraphicsContext::DrawLineTo() */
   881 	{
   882 	iPimpl->WriteAnyPendingStateChanges();
   883 	WritePoint(aPoint,EWsGcOpDrawTo);
   884 	}
   885 
   886 EXPORT_C void CWindowGc::DrawLineBy(const TPoint &aPoint)
   887 /** Draws a straight line relative to the current internal drawing position, using 
   888 a vector.
   889 
   890 The start point of the line is the current internal drawing position. The 
   891 vector aVector is added to the internal drawing position to give the end point 
   892 of the line
   893 
   894 @param aPoint The vector to add to the current internal drawing position, 
   895 giving the end point of the line. 
   896 @see CGraphicsContext::DrawLineBy() */
   897 	{
   898 	iPimpl->WriteAnyPendingStateChanges();
   899 	WritePoint(aPoint,EWsGcOpDrawBy);
   900 	}
   901 
   902 void CWindowGc::doDrawPolyLine(const CArrayFix<TPoint> *aPointArray, const TPoint* aPointList,TInt aNumPoints)
   903 	{
   904 	TWsGcOpcodes opcode=EWsGcOpDrawPolyLine;
   905 	TWsGcCmdDrawPolyLine polyLine;
   906 	TInt maxBufLen=(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(polyLine))/sizeof(TPoint);
   907 	TInt sent=0;
   908 	while(sent<aNumPoints)
   909 		{
   910 		TInt availableLen;
   911 		const TPoint *ptr;
   912 		if (aPointArray)
   913 			{
   914 			ptr=&(*aPointArray)[sent];
   915 			availableLen=aPointArray->End(sent)-ptr;
   916 			}
   917 		else
   918 			{
   919 			ptr=aPointList+sent;
   920 			availableLen=aNumPoints-sent;
   921 			}
   922 		polyLine.numPoints=Min(availableLen,maxBufLen);
   923 		sent+=polyLine.numPoints;
   924 		polyLine.more=(sent!=aNumPoints);
   925 		Write(&polyLine,sizeof(polyLine),ptr,polyLine.numPoints*sizeof(TPoint),opcode);
   926 		polyLine.last=ptr[polyLine.numPoints-1];
   927 		opcode=EWsGcOpDrawPolyLineContinued;
   928 		} 
   929 	}
   930 
   931 EXPORT_C void CWindowGc::DrawPolyLine(const TPoint* aPointList,TInt aNumPoints)
   932 /** Draws a polyline using points in a list. 
   933 
   934 A polyline is a series of concatenated straight lines joining a set of points.
   935 
   936 @param aPointList Pointer to a list of points on the polyline. 
   937 @param aNumPoints The number of points in the point list. 
   938 @see CGraphicsContext::DrawPolyLine() */
   939 	{
   940 	iPimpl->WriteAnyPendingStateChanges();
   941 	doDrawPolyLine(NULL,aPointList,aNumPoints);
   942 	}
   943 
   944 EXPORT_C void CWindowGc::DrawPolyLine(const CArrayFix<TPoint> *aPointArray)
   945 /** Draws a polyline using points in an array. 
   946 
   947 A polyline is a series of concatenated straight lines joining a set of points.
   948 
   949 @param aPointArray An array containing the points on the polyline. 
   950 @see CGraphicsContext::DrawPolyLine() */
   951 	{
   952    	iPimpl->WriteAnyPendingStateChanges();
   953 	doDrawPolyLine(aPointArray,NULL,aPointArray->Count());
   954 	}
   955 
   956 TInt CWindowGc::doDrawPolygon(const CArrayFix<TPoint> *aPointArray,const TPoint* aPointList,TInt aNumPoints,TFillRule aFillRule)
   957 	{
   958 	if (aNumPoints<=0)
   959 		return(KErrNone);
   960 	TWsGcCmdSegmentedDrawPolygonData polyData;
   961 	polyData.index=0;
   962 	TInt maxBufLen=(iBuffer->BufferSize()-EPolygonMaxHeaderSize)/sizeof(TPoint);
   963 	FOREVER
   964 		{
   965 		const TPoint *ptr;
   966 		TInt availableLen;
   967 		if (aPointArray)
   968 			{
   969 			ptr=&(*aPointArray)[polyData.index];
   970 			availableLen=aPointArray->End(polyData.index)-ptr;
   971 			}
   972 		else
   973 			{
   974 			ptr=aPointList+polyData.index;
   975 			availableLen=aNumPoints-polyData.index;
   976 			}
   977 		polyData.numPoints=Min(availableLen,maxBufLen);
   978 		if (polyData.index==0)	// First time around
   979 			{
   980 			if (polyData.numPoints==aNumPoints)	// Can it be done in one go?
   981 				{
   982 				TWsGcCmdDrawPolygon drawPolygon;
   983 				drawPolygon.numPoints=aNumPoints;
   984 				drawPolygon.fillRule=aFillRule;
   985 				Write(&drawPolygon,sizeof(drawPolygon),ptr,aNumPoints*sizeof(TPoint),EWsGcOpDrawPolygon);
   986 				break;
   987 				}
   988 			TWsGcCmdStartSegmentedDrawPolygon start;
   989 			start.totalNumPoints=aNumPoints;
   990 			TInt err=WriteReply(&start,sizeof(start),EWsGcOpStartSegmentedDrawPolygon);
   991 			if (err!=KErrNone)
   992 				return(err);
   993 			}
   994 		Write(&polyData,sizeof(polyData),ptr,polyData.numPoints*sizeof(TPoint),EWsGcOpSegmentedDrawPolygonData);
   995 		polyData.index+=polyData.numPoints;
   996 		if (polyData.index==aNumPoints)
   997 			{
   998 			TWsGcCmdDrawSegmentedPolygon draw;
   999 			draw.fillRule=aFillRule;
  1000 			Write(&draw,sizeof(draw),EWsGcOpDrawSegmentedPolygon);
  1001 			break;
  1002 			}
  1003 		}
  1004 	return(KErrNone);
  1005 	}
  1006 
  1007 EXPORT_C TInt CWindowGc::DrawPolygon(const TPoint* aPointList,TInt aNumPoints,TFillRule aFillRule)
  1008 /** Draws and fills a polygon using points defined in a list.
  1009 
  1010 The first TPoint in the list defines the start of the first side of the polygon. 
  1011 The second TPoint defines the second vertex (the end point of the first side 
  1012 and the start point of the second side) and so on. The final side of the polygon 
  1013 is drawn using the last TPoint from the array or list, and the line drawn 
  1014 to the start point of the first side.
  1015 
  1016 Self-crossing polygons can be filled according to one of two rules, TFillRule::EAlternate 
  1017 (the default), or TFillRule::EWinding. To explain the difference between these 
  1018 rules, the concept of a winding number needs to be introduced. The area outside 
  1019 any of the loops of the polygon has a winding number of zero, and is never 
  1020 filled. An inside a loop which is bounded by an area with winding number 0 
  1021 has a winding number of 1. If an area is within a loop that is bounded by 
  1022 an area with winding number 1, e.g. a loop within a loop, has a winding number 
  1023 of 2, and so on.
  1024 
  1025 The filling of a polygon proceeds according to this algorithm:
  1026 
  1027 If aFillRule is TFillRule::EAlternate (default) and it has an odd winding 
  1028 number, then fill the surrounding area.
  1029 
  1030 If aFillRule is TFillRule::EWinding and it has a winding number greater than 
  1031 zero, then fill the surrounding area.
  1032 
  1033 This function always causes a flush of the window server buffer.
  1034 
  1035 @param aPointList Pointer to a list of points, specifying the vertices of 
  1036 the polygon. 
  1037 @param aNumPoints The number of points in the vertex list 
  1038 @param aFillRule Either TFillRule::EAlternate (the default) or TFillRule::EWinding. 
  1039 @return KErrNone if successful, otherwise another of the system-wide error 
  1040 codes. 
  1041 @see CGraphicsContext::DrawPolygon() */
  1042 	{
  1043 	iPimpl->WriteAnyPendingStateChanges();
  1044 	return(doDrawPolygon(NULL,aPointList,aNumPoints,aFillRule));
  1045 	}
  1046 
  1047 EXPORT_C TInt CWindowGc::DrawPolygon(const CArrayFix<TPoint> *aPointArray,TFillRule aFillRule)
  1048 /** Draws and fills a polygon using points defined in an array.
  1049 
  1050 The first TPoint in the array defines the start of the first side of the polygon. 
  1051 The second TPoint defines the second vertex (the end point of the first side 
  1052 and the start point of the second side) and so on. The final side of the polygon 
  1053 is drawn using the last TPoint from the array or list, and the line drawn 
  1054 to the start point of the first side.
  1055 
  1056 Self-crossing polygons can be filled according to one of two rules, TFillRule::EAlternate 
  1057 (the default), or TFillRule::EWinding. To explain the difference between these 
  1058 rules, the concept of a winding number needs to be introduced. The area outside 
  1059 any of the loops of the polygon has a winding number of zero, and is never 
  1060 filled. An inside a loop which is bounded by an area with winding number 0 
  1061 has a winding number of 1. If an area is within a loop that is bounded by 
  1062 an area with winding number 1, e.g. a loop within a loop, has a winding number 
  1063 of 2, and so on.
  1064 
  1065 The filling of a polygon proceeds according to this algorithm:
  1066 
  1067 If aFillRule is TFillRule::EAlternate (default) and it has an odd winding 
  1068 number, then fill the surrounding area.
  1069 
  1070 If aFillRule is TFillRule::EWinding and it has a winding number greater than 
  1071 zero, then fill the surrounding area.
  1072 
  1073 This function always causes a flush of the window server buffer.
  1074 
  1075 @param aPointArray An array of points, specifying the vertices of the polygon. 
  1076 @param aFillRule Either TFillRule::EAlternate (the default) or TFillRule::EWinding. 
  1077 @return KErrNone if successful, otherwise another of the system-wide error 
  1078 codes. 
  1079 @see CGraphicsContext::DrawPolygon() */
  1080 	{
  1081 	iPimpl->WriteAnyPendingStateChanges();
  1082 	return(doDrawPolygon(aPointArray,NULL,aPointArray->Count(),aFillRule));
  1083 	}
  1084 
  1085 void CWindowGc::DrawArcOrPie(const TRect &aRect,const TPoint &aStart,const TPoint &aEnd, TInt aOpcode)
  1086 	{
  1087 	iPimpl->WriteAnyPendingStateChanges();
  1088 	TWsGcCmdDrawArcOrPie cmd(aRect,aStart,aEnd);
  1089 	Write(&cmd,sizeof(cmd),aOpcode);
  1090 	}
  1091 
  1092 EXPORT_C void CWindowGc::DrawArc(const TRect &aRect,const TPoint &aStart,const TPoint &aEnd)
  1093 /** Draws an arc (a portion of an ellipse).
  1094 
  1095 The point aStart is used to define one end of a line from the geometric centre 
  1096 of the ellipse. The point of intersection between this line and the ellipse 
  1097 defines the start point of the arc. The point aEnd is used to define one end 
  1098 of a second line from the geometric centre of the ellipse. The point of intersection 
  1099 between this line and the ellipse defines the end point of the arc. The pixels 
  1100 at both the start point and the end point are drawn.
  1101 
  1102 The arc itself is the segment of the ellipse in an anti-clockwise direction 
  1103 from the start point to the end point.
  1104 
  1105 Notes
  1106 
  1107 A rectangle is used in the construction of the ellipse of which the arc is 
  1108 a segment. This rectangle is passed as an argument of type TRect.
  1109 
  1110 A wide line arc is drawn with the pixels distributed either side of a true 
  1111 ellipse, in such a way that the outer edge of the line would touch the edge 
  1112 of the construction rectangle. In other words, the ellipse used to construct 
  1113 it is slightly smaller than that for a single pixel line size.
  1114 
  1115 If aStart or aEnd are the ellipse centre then the line that defines the start/end 
  1116 of the arc defaults to one extending vertically above the centre point.
  1117 
  1118 If aStart and aEnd are the same point, or points on the same line through 
  1119 the ellipse centre then a complete unfilled ellipse is drawn.
  1120 
  1121 Line drawing is subject to pen colour, width and style and draw mode
  1122 
  1123 @param aRect The rectangle in which to draw the ellipse (of which the arc is 
  1124 a segment).
  1125 @param aStart A point to define the start of the arc.
  1126 @param aEnd A point to define the end of the arc.
  1127 @see CGraphicsContext::DrawArc() */
  1128 	{
  1129 	iPimpl->WriteAnyPendingStateChanges();
  1130 	DrawArcOrPie(aRect,aStart,aEnd,EWsGcOpDrawArc);
  1131 	}
  1132 
  1133 EXPORT_C void CWindowGc::DrawPie(const TRect &aRect,const TPoint &aStart,const TPoint &aEnd)
  1134 /** Draws and fills a pie-shaped slice of an ellipse.
  1135 
  1136 Outlines are subject to the current pen colour, width, style and draw mode. 
  1137 Set the pen to ENullPen for no outline. The fill is subject to brush style 
  1138 (colour, hash or pattern), the origin and the current drawing mode. Set the 
  1139 brush to ENullBrush for no fill.
  1140 
  1141 The point aStart is used to define one end of a line to the centre of the 
  1142 ellipse. The point of intersection between this line and the ellipse defines 
  1143 the start point of the arc bounding the pie slice. The point aEnd is used 
  1144 to define one end of a second line to the centre of the ellipse. The point 
  1145 of intersection between this line and the ellipse defines the end point of 
  1146 the arc bounding the pie slice. The pixels at the end point are not drawn.
  1147 
  1148 The pie slice itself is the area bounded by: the arc of the ellipse in an 
  1149 anticlockwise direction from the start point to the end point; the straight 
  1150 line from the start point from the geometric centre of the ellipse; the 
  1151 straight line from the end point from the geometric centre of the ellipse.
  1152 
  1153 The line drawn by the pen goes inside the rectangle given by the aRect argument.
  1154 
  1155 Notes:
  1156 
  1157 A rectangle is used in the construction of the pie slice. This rectangle is 
  1158 passed as an argument of type TRect. The curved edge of the pie slice is an 
  1159 arc of an ellipse constructed within the rectangle.
  1160 
  1161 A wide line edged pie slice has the arc drawn with the pixels distributed 
  1162 either side of a true ellipse. This is done in such a way that the outer edge 
  1163 of the line would touch the edge of the construction rectangle. In other words, 
  1164 the ellipse used to construct it is slightly smaller than that for a single 
  1165 pixel line size.
  1166 
  1167 If aStart or aEnd are the ellipse centre then the line that defines the start/end 
  1168 of the arc defaults to one extending vertically above the centre point.
  1169 
  1170 If aStart and aEnd are the same point, or points on the same line through 
  1171 the ellipse centre then a complete filled ellipse is drawn. A line is also 
  1172 drawn from the edge to the ellipse centre.
  1173 
  1174 @param aRect A rectangle in which to draw the ellipse bounding the pie slice 
  1175 @param aStart A point to define the start of the pie slice 
  1176 @param aEnd A point to define the end of the pie slice 
  1177 @see CGraphicsContext::DrawPie() */
  1178 	{
  1179 	iPimpl->WriteAnyPendingStateChanges();
  1180 	DrawArcOrPie(aRect,aStart,aEnd,EWsGcOpDrawPie);
  1181 	}
  1182 
  1183 EXPORT_C void CWindowGc::DrawEllipse(const TRect &aRect)
  1184 /** Draws and fills an ellipse.
  1185 
  1186 The ellipse is drawn inside the rectangle defined by the aRect argument. Any 
  1187 TRect that has odd pixel dimensions, has the bottom right corner trimmed to 
  1188 give even pixel dimensions before the ellipse is constructed.
  1189 
  1190 The column and row of pixels containing the bottom right co-ordinate of the 
  1191 aRect argument are not part of the rectangle.
  1192 
  1193 Note: a wide outline ellipse is drawn with the pixels distributed either side of 
  1194 a true ellipse, in such a way that the outer edge of the line touches the 
  1195 edge of the construction rectangle. In other words, the ellipse used to construct 
  1196 it is smaller than that for a single pixel line size.
  1197 
  1198 @param aRect The rectangle in which to draw the ellipse 
  1199 @see CGraphicsContext::DrawEllipse() */
  1200 	{
  1201 	iPimpl->WriteAnyPendingStateChanges();
  1202 	WriteRect(aRect,EWsGcOpDrawEllipse);
  1203 	}
  1204 
  1205 EXPORT_C void CWindowGc::DrawRect(const TRect &aRect)
  1206 /** Draws and fills a rectangle. 
  1207 
  1208 The rectangle's border is drawn with the pen, and it is filled using the brush.
  1209 
  1210 @param aRect The rectangle to be drawn. 
  1211 @see CGraphicsContext::DrawRect() */
  1212 	{
  1213 	iPimpl->WriteAnyPendingStateChanges();
  1214 	WriteRect(aRect,EWsGcOpDrawRect);
  1215 	}
  1216 
  1217 EXPORT_C void CWindowGc::DrawRoundRect(const TRect &aRect,const TSize &aEllipse)
  1218 /** Draws and fills a rectangle with rounded corners.
  1219 
  1220 The rounded corners are each constructed as an arc of an ellipse. The dimensions 
  1221 of each corner (corner size and corner height) are given by aEllipse. See 
  1222 DrawArc() for a description of arc construction.
  1223 
  1224 The line drawn by the pen (if any) goes inside the rectangle given by the 
  1225 TRect argument.
  1226 
  1227 Notes:
  1228 
  1229 Dotted and dashed pen styles cannot be used for the outline of a rounded rectangle.
  1230 
  1231 If either corner size dimension is greater than half the corresponding rectangle 
  1232 length, the corner size dimension is reduced to half the rectangle size.
  1233 
  1234 @param aRect The rectangle to be drawn. 
  1235 @param aEllipse The dimensions of each corner. 
  1236 @see CGraphicsContext::DrawRoundRect() */
  1237 	{
  1238 	iPimpl->WriteAnyPendingStateChanges();
  1239 	TWsGcCmdDrawRoundRect drawRoundRect(aRect,aEllipse);
  1240 	Write(&drawRoundRect,sizeof(drawRoundRect),EWsGcOpDrawRoundRect);
  1241 	}
  1242 
  1243 EXPORT_C void CWindowGc::DrawBitmap(const TPoint &aTopLeft, const CFbsBitmap *aDevice)
  1244 /** Draws a bitmap at a specified point. 
  1245 
  1246 The function does a compress/stretch based on its internally stored size in 
  1247 twips. Note that if the twips value of the bitmap is not set then nothing 
  1248 is drawn (this is the default situation).
  1249 
  1250 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1251 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1252 At some point later WSERV may need to draw that window again and it will just replay the 
  1253 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1254 WSERV will effectively draw a different bitmap when it replays the commands.
  1255 
  1256 Note: this member function uses the bitmap's size in twips and does a stretch/compress 
  1257 blit using a linear DDA.
  1258 
  1259 @param aTopLeft The point where the top left pixel of the bitmap is to be 
  1260 drawn 
  1261 @param aDevice The source bitmap. 
  1262 @see CGraphicsContext::DrawBitmap() */
  1263 	{
  1264 	iPimpl->WriteAnyPendingStateChanges();
  1265 	TWsGcCmdDrawBitmap drawBitmap(aTopLeft,aDevice->Handle());
  1266 	Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap);
  1267 	AddToBitmapArray(aDevice->Handle());
  1268 	}
  1269 
  1270 EXPORT_C void CWindowGc::DrawBitmap(const TRect &aDestRect, const CFbsBitmap *aDevice)
  1271 /** Draws a bitmap in a rectangle.
  1272 
  1273 The bitmap is compressed/stretched to fit the specified rectangle. Note that 
  1274 if the twips value of the bitmap is not set then nothing is drawn (this is 
  1275 the default situation).
  1276 
  1277 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1278 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1279 At some point later WSERV may need to draw that window again and it will just replay the 
  1280 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1281 WSERV will effectively draw a different bitmap when it replays the commands.
  1282 
  1283 Notes: this member function uses the bitmap's size in pixels and does a stretch/compress 
  1284 blit using a linear DDA.
  1285 
  1286 @param aDestRect The rectangle within which the bitmap is to be drawn. 
  1287 @param aDevice The source bitmap. 
  1288 @see CGraphicsContext::DrawBitmap() */
  1289 	{
  1290 	iPimpl->WriteAnyPendingStateChanges();
  1291 	TWsGcCmdDrawBitmap2 drawBitmap(aDestRect,aDevice->Handle());
  1292 	Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap2);
  1293 	AddToBitmapArray(aDevice->Handle());
  1294 	}
  1295 
  1296 EXPORT_C void CWindowGc::DrawBitmap(const TRect &aDestRect, const CFbsBitmap *aDevice, const TRect &aSourceRect)
  1297 /** Draws a specified rectangle from a bitmap into another rectangle.
  1298 
  1299 The function compresses/stretches the specified rectangle from the bitmap 
  1300 to fit the destination rectangle. Note that if the twips value of the bitmap 
  1301 is not set then nothing is drawn (this is the default situation).
  1302 
  1303 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1304 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1305 At some point later WSERV may need to draw that window again and it will just replay the 
  1306 stored commands including the draw bitmap. However, if the client has changed the content of the bitmap,
  1307 WSERV will effectively draw a different bitmap when it replays the commands.
  1308 
  1309 Note: this member function uses rectangle sizes in pixels and does a stretch/compress 
  1310 blit using a linear DDA.
  1311 
  1312 @param aDestRect The rectangle within which the bitmap is to be drawn. 
  1313 @param aDevice A source bitmap. 
  1314 @param aSourceRect The rectangle in the source bitmap that is copied to the 
  1315 destination rectangle. 
  1316 @see CGraphicsContext::DrawBitmap() */
  1317 	{
  1318 	iPimpl->WriteAnyPendingStateChanges();
  1319 	TWsGcCmdDrawBitmap3 drawBitmap(aDestRect,aDevice->Handle(),aSourceRect);
  1320 	Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap3);
  1321 	AddToBitmapArray(aDevice->Handle());
  1322 	}
  1323 
  1324 /** Draws a specified rectangle from a bitmap and its mask into another rectangle.
  1325 
  1326 The function compresses/stretches the specified rectangle from the bitmap 
  1327 to fit the destination rectangle. 
  1328 The mask bitmap can be used as either a positive or negative mask. Masked 
  1329 pixels are not mapped to the destination rectangle.
  1330 
  1331 A black and white (binary) mask bitmap is used. With aInvertMask=EFalse, black 
  1332 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1333 being transferred to the destination rectangle. With aInvertMask=ETrue, white 
  1334 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1335 being transferred to the destination rectangle.
  1336 
  1337 If mask bitmap's display mode is EColor256, the function does AplhaBlending
  1338 and ignores aInvertMask parameter.
  1339 
  1340 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1341 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1342 At some point later WSERV may need to draw that window again and it will just replay the 
  1343 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1344 WSERV will effectively draw a different bitmap when it replays the commands.
  1345 
  1346 Note: this member function uses rectangle sizes in pixels and does a stretch/compress 
  1347 blit using a linear DDA.
  1348 
  1349 @param aDestRect The rectangle within which the masked bitmap is to be drawn. 
  1350 @param aBitmap A source bitmap. 
  1351 @param aSourceRect The rectangle in the source bitmap that is copied to the 
  1352 destination rectangle.
  1353 @param aMaskBitmap A mask bitmap. 
  1354 @param aInvertMask If false, a source pixel that is masked by a black pixel 
  1355 is not transferred to the destination rectangle. If true, then a source pixel 
  1356 that is masked by a white pixel is not transferred to the destination rectangle. */
  1357 EXPORT_C void CWindowGc::DrawBitmapMasked(const TRect& aDestRect, const CFbsBitmap* aBitmap, const TRect& aSourceRect, const CFbsBitmap* aMaskBitmap, TBool aInvertMask)
  1358 	{
  1359 	iPimpl->WriteAnyPendingStateChanges();
  1360 	TWsGcCmdDrawBitmapMasked drawBitmap(aDestRect,aBitmap->Handle(),aSourceRect,aMaskBitmap->Handle(),aInvertMask);
  1361 	Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmapMasked);
  1362 	AddToBitmapArray(aBitmap->Handle());
  1363 	AddToBitmapArray(aMaskBitmap->Handle());
  1364 	}
  1365 
  1366 EXPORT_C void CWindowGc::DrawBitmapMasked(const TRect& aDestRect, const CWsBitmap* aBitmap, const TRect& aSourceRect, const CWsBitmap* aMaskBitmap, TBool aInvertMask)
  1367 /** Draws a specified rectangle from a wserv bitmap and its mask into 
  1368 another rectangle.
  1369 
  1370 The function compresses/stretches the specified rectangle from the bitmap 
  1371 to fit the destination rectangle. 
  1372 The mask bitmap can be used as either a positive or negative mask. Masked 
  1373 pixels are not mapped to the destination rectangle.
  1374 
  1375 A black and white (binary) mask bitmap is used. With aInvertMask=EFalse, black 
  1376 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1377 being transferred to the destination rectangle. With aInvertMask=ETrue, white 
  1378 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1379 being transferred to the destination rectangle.
  1380 
  1381 If mask bitmap's display mode is EColor256, the function does AplhaBlending
  1382 and ignores aInvertMask parameter.
  1383 
  1384 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1385 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1386 At some point later WSERV may need to draw that window again and it will just replay the 
  1387 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1388 WSERV will effectively draw a different bitmap when it replays the commands.
  1389 
  1390 Note: this member function uses rectangle sizes in pixels and does a stretch/compress 
  1391 blit using a linear DDA.
  1392 
  1393 @param aDestRect The rectangle within which the masked bitmap is to be drawn. 
  1394 @param aBitmap A source wserv bitmap. 
  1395 @param aSourceRect The rectangle in the source bitmap that is copied to the 
  1396 destination rectangle.
  1397 @param aMaskBitmap A mask wserv bitmap. 
  1398 @param aInvertMask If false, a source pixel that is masked by a black pixel 
  1399 is not transferred to the destination rectangle. If true, then a source pixel 
  1400 that is masked by a white pixel is not transferred to the destination rectangle. */
  1401 	{
  1402 	iPimpl->WriteAnyPendingStateChanges();
  1403 	TWsGcCmdDrawBitmapMasked drawBitmap(aDestRect,aBitmap->WsHandle(),aSourceRect,aMaskBitmap->WsHandle(),aInvertMask);
  1404 	Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpWsDrawBitmapMasked);
  1405 	AddToBitmapArray(aBitmap->Handle());
  1406 	AddToBitmapArray(aMaskBitmap->Handle());
  1407 	}
  1408 EXPORT_C void CWindowGc::DrawText(const TDesC &aBuf, const TPoint &aPos)
  1409 /** Draws horizontal text with no surrounding box. 
  1410 
  1411 The appearance of the text is subject to the drawing mode, the font, pen colour, 
  1412 word justification and character justification. 
  1413 
  1414 A panic occurs if this function is called when there is no font: see UseFont().
  1415 
  1416 @param aBuf The string to write. 
  1417 @param aPos The point specifying the position of the baseline at the left 
  1418 end of the text.
  1419 @see CGraphicsContext::DrawText() */
  1420 	{
  1421 	iPimpl->WriteAnyPendingStateChanges();
  1422 	TWsGcCmdDrawText printText(aPos,aBuf.Length());
  1423     __ASSERT_ALWAYS(((aBuf.Size()+sizeof(printText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected));    
  1424 	WriteTextCommand(&printText,sizeof(printText),aBuf,EWsGcOpDrawText,EWsGcOpDrawTextPtr);
  1425 	}
  1426 
  1427 EXPORT_C void CWindowGc::DrawText(const TDesC &aBuf,const TRect &aBox,TInt aBaselineOffset,TTextAlign aHoriz,TInt aLeftMrg)
  1428 /** Draws horizontal text within a cleared box.
  1429 
  1430 The appearance of the text is subject to the drawing mode, the font, pen colour, 
  1431 word justification and character justification. It is also subject to the 
  1432 background brush (set brush to ENullBrush for no effect on background).
  1433 
  1434 A panic occurs if this function is called when there is no font: see UseFont().
  1435 
  1436 Note: the text is clipped to the box. You must ensure that the specified string 
  1437 is not too large.
  1438 
  1439 @param aBuf The text to write.
  1440 @param aBox The box to draw the text in. 
  1441 @param aBaselineOffset An offset from the top of the box to the text baseline. 
  1442 Note that the baseline is the line on which letters sit, for instance below r, s, t, and 
  1443 above the tail of q, and y. 
  1444 @param aHoriz The text alignment mode (default is left, rather than centre 
  1445 or right). 
  1446 @param aLeftMrg The left margin for left-aligned text, or the right margin 
  1447 for right-aligned text (default is zero). 
  1448 @see CGraphicsContext::DrawText() */
  1449 	{
  1450 	iPimpl->WriteAnyPendingStateChanges();
  1451 	if (aBuf.Size()<(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(TWsGcCmdBoxTextOptimised2))) 
  1452 		{
  1453 		if (aHoriz==ELeft && aLeftMrg==0)
  1454 			{
  1455 			TWsGcCmdBoxTextOptimised1 boxTextOpt1(aBox,aBaselineOffset,aBuf.Length());
  1456 			__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxTextOpt1))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1457 			Write(&boxTextOpt1,sizeof(boxTextOpt1),aBuf.Ptr(),aBuf.Size(),EWsGcOpDrawBoxTextOptimised1);
  1458 			}
  1459 		else
  1460 			{
  1461 			TWsGcCmdBoxTextOptimised2 boxTextOpt2(aBox,aBaselineOffset,aHoriz,aLeftMrg,aBuf.Length());
  1462 			__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxTextOpt2))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1463 			Write(&boxTextOpt2,sizeof(boxTextOpt2),aBuf.Ptr(),aBuf.Size(),EWsGcOpDrawBoxTextOptimised2);
  1464 			}
  1465 		}
  1466 	else
  1467 		{
  1468 		TWsGcCmdBoxText boxText(aBox,aBaselineOffset,aHoriz,aLeftMrg,aBuf.Length(),iPimpl->iFont->TextWidthInPixels(aBuf));
  1469 		__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1470 		WriteTextCommand(&boxText,sizeof(boxText),aBuf,EWsGcOpDrawBoxText,EWsGcOpDrawBoxTextPtr);
  1471 		}
  1472 	}
  1473 
  1474 TInt CWindowGc::APIExDrawText(const TDesC& aBuf,const TTextParameters* aParam,const TPoint& aPos)
  1475 	{
  1476 	iPimpl->WriteAnyPendingStateChanges();
  1477 	TWsGcCmdDrawTextInContext printTextInContext(aPos,aBuf.Length(),aParam->iStart,aParam->iEnd);
  1478 	__ASSERT_ALWAYS(((aBuf.Size()+sizeof(printTextInContext))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1479 	WriteTextCommand(&printTextInContext,sizeof(printTextInContext),aBuf,EWsGcOpDrawTextInContext,EWsGcOpDrawTextInContextPtr);
  1480 	return KErrNone;
  1481 	}
  1482 	
  1483 TInt CWindowGc::APIExDrawText(const TDesC& aBuf,const TTextParameters* aParam,const TRect& aBox,TInt aBaselineOffset,TTextAlign aHoriz,TInt aLeftMrg)
  1484 	{
  1485 	iPimpl->WriteAnyPendingStateChanges();
  1486 	if (aBuf.Size()<(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(TWsGcCmdBoxTextInContextOptimised2))) 
  1487 		{
  1488 		if (aHoriz==ELeft && aLeftMrg==0)
  1489 			{
  1490 			TWsGcCmdBoxTextInContextOptimised1 boxTextOpt1(aBox,aBaselineOffset,aBuf.Length(),aParam->iStart,aParam->iEnd);
  1491 			__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxTextOpt1))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1492 			Write(&boxTextOpt1,sizeof(boxTextOpt1),aBuf.Ptr(),aBuf.Size(),EWsGcOpDrawBoxTextInContextOptimised1);
  1493 			}
  1494 		else
  1495 			{
  1496 			TWsGcCmdBoxTextInContextOptimised2 boxTextOpt2(aBox,aBaselineOffset,aHoriz,aLeftMrg,aBuf.Length(),aParam->iStart,aParam->iEnd);
  1497 			__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxTextOpt2))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1498 			Write(&boxTextOpt2,sizeof(boxTextOpt2),aBuf.Ptr(),aBuf.Size(),EWsGcOpDrawBoxTextInContextOptimised2);
  1499 			}
  1500 		}
  1501 	else
  1502 		{
  1503 		TWsGcCmdBoxTextInContext boxText(aBox,aBaselineOffset,aHoriz,aLeftMrg,aBuf.Length(),iPimpl->iFont->TextWidthInPixels(aBuf),aParam->iStart,aParam->iEnd);
  1504 		__ASSERT_ALWAYS(((aBuf.Size()+sizeof(boxText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1505 		WriteTextCommand(&boxText,sizeof(boxText),aBuf,EWsGcOpDrawBoxTextInContext,EWsGcOpDrawBoxTextInContextPtr);
  1506 		}
  1507 	return KErrNone;
  1508 	}
  1509 EXPORT_C void CWindowGc::DrawTextVertical(const TDesC& aText,const TPoint& aPos,TBool aUp)
  1510 /** Draws vertical text in the specified direction.
  1511 
  1512 A panic occurs if this function is called when there is no font: see UseFont().
  1513 
  1514 @param aText The text to be drawn. 
  1515 @param aPos Point of origin of the text baseline. 
  1516 @param aUp Direction. ETrue for up, EFalse for down. */
  1517 	{
  1518 	iPimpl->WriteAnyPendingStateChanges();
  1519 	TWsGcCmdDrawTextVertical printText(aPos,aText.Length(),aUp);
  1520 	__ASSERT_ALWAYS(((aText.Size()+sizeof(printText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1521 	WriteTextCommand(&printText,sizeof(printText),aText,EWsGcOpDrawTextVertical,EWsGcOpDrawTextVerticalPtr);
  1522 	}
  1523 
  1524 EXPORT_C void CWindowGc::DrawTextVertical(const TDesC& aText,const TRect& aBox,TInt aBaselineOffset,TBool aUp,TTextAlign aVert,TInt aMargin)
  1525 /** Draws text vertically in the specified direction, within a box of the specified 
  1526 size.
  1527 
  1528 A panic occurs if this function is called when there is no font: see UseFont().
  1529 
  1530 @param aText The text to be drawn. 
  1531 @param aBox The bounding box within which the text should be drawn, and which 
  1532 it is clipped to.
  1533 @param aBaselineOffset The height of the top of the characters from their text 
  1534 baseline. 
  1535 @param aUp The direction. ETrue for up, EFalse for down.
  1536 @param aVert The text alignment. 
  1537 @param aMargin The margin. */
  1538 	{
  1539 	iPimpl->WriteAnyPendingStateChanges();
  1540 	TWsGcCmdBoxTextVertical boxText(aBox);
  1541 	boxText.baselineOffset=aBaselineOffset;
  1542 	boxText.up=aUp;
  1543 	boxText.vert=aVert;
  1544 	boxText.margin=aMargin;
  1545 	boxText.length=aText.Length();
  1546 	boxText.width=iPimpl->iFont->TextWidthInPixels(aText);
  1547 	__ASSERT_ALWAYS(((aText.Size()+sizeof(boxText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1548 	WriteTextCommand(&boxText,sizeof(boxText),aText,EWsGcOpDrawBoxTextVertical,EWsGcOpDrawBoxTextVerticalPtr);
  1549 	}
  1550 
  1551 TInt CWindowGc::APIExDrawTextVertical(const TDesC& aText,const TTextParameters* aParam,const TPoint& aPos,TBool aUp)
  1552 	{
  1553 	iPimpl->WriteAnyPendingStateChanges();
  1554 	TWsGcCmdDrawTextInContextVertical printText(aPos,aText.Length(),aUp,aParam->iStart,aParam->iEnd);
  1555 	__ASSERT_ALWAYS(((aText.Size()+sizeof(printText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1556 	WriteTextCommand(&printText,sizeof(printText),aText,EWsGcOpDrawTextInContextVertical,EWsGcOpDrawTextInContextVerticalPtr);
  1557 	return KErrNone;
  1558 	}
  1559 
  1560 TInt CWindowGc::APIExDrawTextVertical(const TDesC& aText,const TTextParameters* aParam,const TRect& aBox,TInt aBaselineOffset,TBool aUp,TTextAlign aVert,TInt aMargin)
  1561 	{
  1562 	iPimpl->WriteAnyPendingStateChanges();
  1563 	TWsGcCmdBoxTextInContextVertical boxText(aBox);
  1564 	boxText.baselineOffset=aBaselineOffset;
  1565 	boxText.up=aUp;
  1566 	boxText.vert=aVert;
  1567 	boxText.margin=aMargin;
  1568 	boxText.length=aText.Length();
  1569 	boxText.width=iPimpl->iFont->TextWidthInPixels(aText);
  1570 	boxText.start = aParam->iStart;
  1571 	boxText.end = aParam->iEnd;
  1572 	__ASSERT_ALWAYS(((aText.Size()+sizeof(boxText))<=EClientBufferMaxSize),Panic(EW32PanicSizeNotExpected)); 
  1573 	WriteTextCommand(&boxText,sizeof(boxText),aText,EWsGcOpDrawBoxTextInContextVertical,EWsGcOpDrawBoxTextInContextVerticalPtr);
  1574 	return KErrNone;
  1575 	}
  1576 
  1577 //========================Extra functions============================
  1578 
  1579 EXPORT_C void CWindowGc::CopyRect(const TPoint &anOffset,const TRect &aRect)
  1580 /** Copies a rectangle from any part of the screen into the window that the gc 
  1581 is active on.
  1582 
  1583 The copy part of the operation applies to the whole rectangle, irrespective 
  1584 of whether or not it within the window, however the "paste" is clipped to 
  1585 the drawing area.
  1586 
  1587 The rectangle is specified in window coordinates (if the top-left of the rectangle 
  1588 is (0,0) then the area of the screen it specifies has its top-left at the 
  1589 top left corner of the window, if it is (-10,-10) then it starts 10 pixels 
  1590 above and to the left of the window). 
  1591 
  1592 Note: shadows in the source rectangle will be copied. None of the area drawn to 
  1593 will gain shadowing (even if the window is already in shadow).
  1594 
  1595 This version of this function is only really suitable for testing.
  1596 
  1597 @param anOffset The offset from the original position to the point where the 
  1598 rectangle is copied. 
  1599 @param aRect The rectangular area to be copied. This is in window co-ordinates, 
  1600 e.g. the top left corner of the window is position (0,0) with respect to the 
  1601 rectangle.
  1602 @see CBitmapContext::CopyRect() */
  1603 	{
  1604 	iPimpl->WriteAnyPendingStateChanges();
  1605 	TWsGcCmdCopyRect copyRect(anOffset,aRect);
  1606 	Write(&copyRect,sizeof(copyRect),EWsGcOpCopyRect);
  1607 	}
  1608 
  1609 EXPORT_C void CWindowGc::BitBlt(const TPoint &aPoint, const CFbsBitmap *aBitmap)
  1610 /** Performs a bitmap block transfer.
  1611 
  1612 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1613 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1614 At some point later WSERV may need to draw that window again and it will just replay the 
  1615 stored commands including the draw bitmap. However, if the client has changed the content of the bitmap,
  1616 WSERV will effectively draw a different bitmap when it replays the commands.
  1617 
  1618 @param aPoint The position for the top left corner of the bitmap. 
  1619 @param aBitmap A memory-resident bitmap. 
  1620 @see CBitmapContext::BitBlt() */
  1621 	{
  1622 	if (aBitmap == NULL || !aBitmap->Handle())
  1623 		return; 
  1624 	iPimpl->WriteAnyPendingStateChanges();
  1625 	TWsGcCmdGdiBlt2 gdiBlit(aPoint,aBitmap->Handle());
  1626 	Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiBlt2);
  1627 	AddToBitmapArray(aBitmap->Handle());
  1628 	}
  1629 
  1630 EXPORT_C void CWindowGc::BitBlt(const TPoint &aDestination,const CFbsBitmap *aBitmap,const TRect &aSource)
  1631 /** Performs a bitmap block transfer of a rectangular piece of a bitmap.
  1632 
  1633 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1634 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1635 At some point later WSERV may need to draw that window again and it will just replay the 
  1636 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1637 WSERV will effectively draw a different bitmap when it replays the commands.
  1638 
  1639 Note: if the rectangle aSource is larger than the bitmap then the bitmap will be padded 
  1640 with white.
  1641 
  1642 @param aDestination The position for the top left corner of the bitmap. 
  1643 @param aBitmap A memory-resident bitmap 
  1644 @param aSource A rectangle defining the piece of the bitmap to be drawn, with 
  1645 co-ordinates relative to the top left corner of the bitmap 
  1646 @see CBitmapContext::BitBlt() */
  1647 	{
  1648 	if (aBitmap == NULL || !aBitmap->Handle())
  1649 		return; 
  1650 	iPimpl->WriteAnyPendingStateChanges();
  1651 	TWsGcCmdGdiBlt3 gdiBlit(aDestination,aBitmap->Handle(),aSource);
  1652 	Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiBlt3);
  1653 	AddToBitmapArray(aBitmap->Handle());
  1654 	}
  1655 
  1656 EXPORT_C void CWindowGc::BitBltMasked(const TPoint& aPoint,const CFbsBitmap* aBitmap,const TRect& aSourceRect,const CFbsBitmap* aMaskBitmap,TBool aInvertMask)
  1657 /** Performs a masked bitmap block transfer of a memory resident source bitmap.
  1658 
  1659 The mask bitmap can be used as either a positive or negative mask. Masked 
  1660 pixels are not mapped to the destination rectangle.
  1661 
  1662 A black and white (binary) mask bitmap is used. With aInvertMask=EFalse, black 
  1663 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1664 being transferred to the destination rectangle. With aInvertMask=ETrue, white 
  1665 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1666 being transferred to the destination rectangle.
  1667 
  1668 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1669 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1670 At some point later WSERV may need to draw that window again and it will just replay the 
  1671 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1672 WSERV will effectively draw a different bitmap when it replays the commands.
  1673 
  1674 @param aPoint A position for the top left corner of the bitmap. 
  1675 @param aBitmap A memory-resident source bitmap. 
  1676 @param aSourceRect A rectangle defining the piece of the bitmap to be drawn, 
  1677 with co-ordinates relative to the top left corner of the bitmap 
  1678 @param aMaskBitmap A mask bitmap. 
  1679 @param aInvertMask If false, a source pixel that is masked by a black pixel 
  1680 is not transferred to the destination rectangle. If true, then a source pixel 
  1681 that is masked by a white pixel is not transferred to the destination rectangle. 
  1682 
  1683 @see CBitmapContext::BitBltMasked() */
  1684 	{
  1685 	if (aBitmap == NULL || !aBitmap->Handle() || aMaskBitmap == NULL || !aMaskBitmap->Handle())
  1686 		return; 
  1687 	iPimpl->WriteAnyPendingStateChanges();
  1688 	TWsGcCmdBltMasked gdiBlitMasked(aPoint,aBitmap->Handle(),aSourceRect,aMaskBitmap->Handle(),aInvertMask);
  1689 	Write(&gdiBlitMasked,sizeof(gdiBlitMasked),EWsGcOpGdiBltMasked);
  1690 	AddToBitmapArray(aBitmap->Handle());
  1691 	AddToBitmapArray(aMaskBitmap->Handle());
  1692 	}
  1693 
  1694 EXPORT_C void CWindowGc::BitBlt(const TPoint &aPoint, const CWsBitmap *aBitmap)
  1695 /** Performs a bitmap block transfer on a bitmap to which the window server already 
  1696 has a handle.
  1697 
  1698 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1699 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1700 At some point later WSERV may need to draw that window again and it will just replay the 
  1701 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1702 WSERV will effectively draw a different bitmap when it replays the commands.
  1703 
  1704 This function should be used in preference to the CFbsBitmap overload if the 
  1705 bitmap is to be used more than once, as it is a lot quicker.
  1706 
  1707 @param aPoint The position for the top left corner of the bitmap. 
  1708 @param aBitmap A window server bitmap. 
  1709 @see CBitmapContext::BitBlt() */
  1710 	{
  1711 	if (aBitmap == NULL || !aBitmap->Handle())
  1712 		return; 
  1713 	iPimpl->WriteAnyPendingStateChanges();
  1714 	TWsGcCmdGdiBlt2 gdiBlit(aPoint,aBitmap->WsHandle());
  1715 	Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiWsBlt2);
  1716 	AddToBitmapArray(aBitmap->Handle());
  1717 	}
  1718 
  1719 EXPORT_C void CWindowGc::BitBlt(const TPoint &aDestination,const CWsBitmap  *aBitmap,const TRect &aSource)
  1720 /** Performs a bitmap block transfer of a rectangular piece of a bitmap to which 
  1721 the window server already has a handle.
  1722 
  1723 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1724 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1725 At some point later WSERV may need to draw that window again and it will just replay the 
  1726 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1727 WSERV will effectively draw a different bitmap when it replays the commands.
  1728 
  1729 This function should be used in preference to the CFbsBitmap overload if the 
  1730 bitmap is to be used more than once, as it is a lot quicker.
  1731 
  1732 Note: if the rectangle aSource is larger than the bitmap then the bitmap will be padded 
  1733 with white.
  1734 	
  1735 @param aDestination The position for the top left corner of the bitmap. 
  1736 @param aBitmap A window server bitmap. 
  1737 @param aSource A rectangle defining the piece of the bitmap to be drawn, with 
  1738 co-ordinates relative to the top left corner of the bitmap 
  1739 @see CBitmapContext::BitBlt() */
  1740 	{
  1741 	if (aBitmap == NULL || !aBitmap->Handle())
  1742 		return; 
  1743 	iPimpl->WriteAnyPendingStateChanges();
  1744 	TWsGcCmdGdiBlt3 gdiBlit(aDestination,aBitmap->WsHandle(),aSource);
  1745 	Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiWsBlt3);
  1746 	AddToBitmapArray(aBitmap->Handle());
  1747 	}
  1748 
  1749 EXPORT_C void CWindowGc::BitBltMasked(const TPoint& aPoint,const CWsBitmap * aBitmap,const TRect& aSourceRect,const CWsBitmap * aMaskBitmap,TBool aInvertMask)
  1750 /** Performs a masked bitmap block transfer of a window server bitmap.
  1751 
  1752 The mask bitmap can be used as either a positive or negative mask. Masked 
  1753 pixels are not mapped to the destination rectangle.
  1754 
  1755 A black and white (binary) mask bitmap is used. With aInvertMask=EFalse, black 
  1756 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1757 being transferred to the destination rectangle. With aInvertMask=ETrue, white 
  1758 pixels in the mask bitmap stop corresponding pixels in the source bitmap from 
  1759 being transferred to the destination rectangle.
  1760 
  1761 This function should be used in preference to the CFbsBitmap overload if the 
  1762 bitmap is to be used more than once, as it is a lot quicker.
  1763 
  1764 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1765 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1766 At some point later WSERV may need to draw that window again and it will just replay the 
  1767 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1768 WSERV will effectively draw a different bitmap when it replays the commands.
  1769 
  1770 @param aPoint A position for the top left corner of the bitmap. 
  1771 @param aBitmap A window server bitmap. 
  1772 @param aSourceRect A rectangle defining the piece of the bitmap to be drawn, 
  1773 with co-ordinates relative to the top left corner of the bitmap. 
  1774 @param aMaskBitmap A window server mask bitmap. 
  1775 @param aInvertMask If false, a source pixel that is masked by a black pixel 
  1776 is not transferred to the destination rectangle. If true, then a source pixel 
  1777 that is masked by a white pixel is not transferred to the destination rectangle. 
  1778 
  1779 @see CBitmapContext::BitBltMasked() */
  1780 	{
  1781 	if (aBitmap == NULL || !aBitmap->Handle() || aMaskBitmap == NULL || !aMaskBitmap->Handle())
  1782 		return;
  1783 	iPimpl->WriteAnyPendingStateChanges();
  1784 	TWsGcCmdBltMasked gdiBlitMasked(aPoint,aBitmap->WsHandle(),aSourceRect,aMaskBitmap->WsHandle(),aInvertMask);
  1785 	Write(&gdiBlitMasked,sizeof(gdiBlitMasked),EWsGcOpGdiWsBltMasked);
  1786 	AddToBitmapArray(aBitmap->Handle());
  1787 	AddToBitmapArray(aMaskBitmap->Handle()); 
  1788 	}
  1789 
  1790 /**
  1791 This method has been deprecated. It is no longer possible to re-map pixel colours
  1792 within a rectangle. Calling it has no effect.
  1793 @param aRect Ignored.
  1794 @param aColors Ignored.
  1795 @param aNumPairs Ignored.
  1796 @param aMapForwards Ignored.
  1797 @deprecated
  1798 */
  1799 EXPORT_C void CWindowGc::MapColors(const TRect& /*aRect*/, const TRgb* /*aColors*/, TInt /*aNumPairs*/, TBool /*aMapForwards*/)
  1800 	{
  1801 	}
  1802 
  1803 EXPORT_C void CWindowGc::Clear(const TRect &aRect)
  1804 /** Clears a rectangular area of a window.
  1805 
  1806 The cleared area is filled with the current brush colour.
  1807 
  1808 @param aRect The rectangle to clear. 
  1809 @see CBitmapContext::Clear() */
  1810 	{
  1811 	iPimpl->WriteAnyPendingStateChanges();
  1812 	WriteRect(aRect,EWsGcOpClearRect);
  1813 	}
  1814 
  1815 EXPORT_C void CWindowGc::Clear()
  1816 /** Clears the whole window.
  1817 
  1818 The cleared area is filled with the current brush colour.
  1819 
  1820 @see CBitmapContext::Clear() */
  1821 	{
  1822 	iPimpl->WriteAnyPendingStateChanges();
  1823 	Write(EWsGcOpClear);
  1824 	}
  1825 
  1826 EXPORT_C void CWindowGc::Reset()
  1827 /** Resets the graphics context to its default settings.
  1828 
  1829 The drawing mode is set to TDrawMode::EDrawModePen (pen and brush colours used as 
  1830 they are); there is no clipping rectangle; the pen settings are black, 
  1831 solid, single pixel size; the brush style is null; no text font is selected.
  1832 
  1833 @see CGraphicsContext::Reset() */
  1834 	{
  1835 	Write(EWsGcOpReset);
  1836 	iPimpl->iFont=NULL;
  1837 	iPimpl->iShadowColor = KDefaultShadowColor;
  1838     iPimpl->ResetPendingState();
  1839     iPimpl->iForceWrite = ETrue; // needed because brush colour set to window background colour in CPlaybackGc::CommandL and CWsGc::SetGcAttribute
  1840 	}
  1841 
  1842 /**
  1843 This method has been deprecated. Dithering is no longer supported. Calling it
  1844 has no effect.
  1845 @param aPoint Ignored.
  1846 @deprecated
  1847 */
  1848 EXPORT_C void CWindowGc::SetDitherOrigin(const TPoint& /*aPoint*/)
  1849 	{
  1850 	}
  1851 
  1852 EXPORT_C void CWindowGc::SetFaded(TBool aFaded)
  1853 /** Sets whether the graphics context is faded.
  1854 
  1855 Fading is used to make a window appear less colourful so that other windows 
  1856 stand out. For example, a window would be faded when a dialogue is displayed 
  1857 in front of it.
  1858 
  1859 @param aFaded ETrue to fade the graphics context, EFalse to unfade it. */
  1860 	{
  1861 	iPimpl->WriteAnyPendingStateChanges();
  1862 	WriteInt(aFaded,EWsGcOpSetFaded);
  1863 	}
  1864 
  1865 EXPORT_C void CWindowGc::SetFadingParameters(TUint8 aBlackMap,TUint8 aWhiteMap)
  1866 /** Sets the fading parameters.
  1867 
  1868 This function allows you to override the map used when drawing with a faded 
  1869 graphics context. However if you draw to a faded window with a faded graphics 
  1870 context, then fading on the graphics context is ignored and it will use the 
  1871 fading of the window.
  1872 
  1873 Fading is used to make a window appear less colourful so that other windows stand 
  1874 out. For example, a window would be faded when a dialogue is displayed 
  1875 in front of it. 
  1876 
  1877 You can either make a faded window closer to white or closer to black. 
  1878 The fading map allows you to over-ride the default fading parameters set in 
  1879 RWsSession::SetDefaultFadingParameters(). 
  1880 
  1881 Fading re-maps colours to fall between the specified black and white map values. 
  1882 If aBlackMap=0 and aWhiteMap=255 then the colours are mapped unchanged. As the 
  1883 values converge, the colours are mapped to a smaller range, so the differences 
  1884 between colours in the faded graphics context decrease. If the values are reversed 
  1885 then the colours are inverted (i.e. where the gc would be black, it is now white). 
  1886 
  1887 @param aBlackMap Black map fading parameter. Unfaded this is 0. 
  1888 @param aWhiteMap White map fading parameter. Unfaded this is 255. 
  1889 @see RWsSession::SetDefaultFadingParameters()
  1890 @see RWindowTreeNode::SetFaded() */
  1891 	{
  1892 	WriteInt(WservEncoding::Encode8BitValues(aBlackMap,aWhiteMap),EWsGcOpSetFadeParams);
  1893 	}
  1894 
  1895 EXPORT_C TInt CWindowGc::AlphaBlendBitmaps(const TPoint& aDestPt, const CFbsBitmap* aSrcBmp, const TRect& aSrcRect,const CFbsBitmap* aAlphaBmp, const TPoint& aAlphaPt)
  1896 /**
  1897 Performs an alpha blending of the source data, aSrcBmp, with the window, using
  1898 the data from aAlphaBmp as an alpha blending factor.
  1899 The formula used is:
  1900 (S * A + W * (255 - A)) / 255, where:
  1901 - S - a pixel from aSrcBmp;
  1902 - W - a pixel from the window;
  1903 - A - a pixel from aAlphaBmp;
  1904 The contents of source and alpha bitmap are preserved.
  1905 The calculated alpha blended pixels are written to the destination - the window image.
  1906 
  1907 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1908 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1909 At some point later WSERV may need to draw that window again and it will just replay the 
  1910 stored commands including the draw bitmap.  However, if the client has changed the content of the bitmap,
  1911 WSERV will effectively draw a different bitmap when it replays the commands.
  1912 
  1913 This method is supported from version 8.1
  1914 @param aDestPt Position in the target the result should be drawn to.
  1915 @param aSrcBmp A pointer to the source bitmap.
  1916 @param aSrcRect The part of the source bitmap that should be used.
  1917 @param aAlphaBmp A pointer to the bitmap used as an alpha blending factor.
  1918 @param aAlphaPt Position of the first pixel in the alpha bitmap that should be used as a source 
  1919                 for the alpha blending. The size of the area is the same as the 
  1920                 source bitmap area - aSrcRect parameter.
  1921 @see CFbsBitGc::AlphaBlendBitmaps()
  1922 */
  1923 	{
  1924    	iPimpl->WriteAnyPendingStateChanges();
  1925 	TWsGcCmdAlphaBlendBitmaps alphaBlend(aDestPt, aSrcBmp->Handle(), aSrcRect, aAlphaBmp->Handle(), aAlphaPt);
  1926 	Write(&alphaBlend,sizeof(alphaBlend),EWsGcOpGdiAlphaBlendBitmaps);
  1927 	AddToBitmapArray(aSrcBmp->Handle());
  1928 	AddToBitmapArray(aAlphaBmp->Handle());
  1929 	return KErrNone;
  1930 	}
  1931 
  1932 EXPORT_C TInt CWindowGc::AlphaBlendBitmaps(const TPoint& aDestPt, const CWsBitmap* aSrcBmp, const TRect& aSrcRect,const CWsBitmap* aAlphaBmp, const TPoint& aAlphaPt)
  1933 /**
  1934 The method performs an alpha blending of the source data, aSrcBmp, with the window, using
  1935 the data from aAlphaBmp as an alpha blending factor.
  1936 For information on how this function works, see the other overload.
  1937 
  1938 Windows that store their redraw commands will only store drawing position and a handle to bitmaps 
  1939 that are drawn in it. The bitmap handle is just a pointer to the bitmap in the FBSERV heap. 
  1940 At some point later WSERV may need to draw that window again and it will just replay the 
  1941 stored commands including the draw bitmap. However, if the client has changed the content of the bitmap,
  1942 WSERV will effectively draw a different bitmap when it replays the commands.
  1943 
  1944 This method is supported from version 8.1
  1945 @param aDestPt Position in the target the result should be drawn to.
  1946 @param aSrcBmp A pointer to the source bitmap.
  1947 @param aSrcRect The part of the source bitmap that should be used.
  1948 @param aAlphaBmp A pointer to the bitmap used as an alpha blending factor.
  1949 @param aAlphaPt Position of the first pixel in the alpha bitmap that should be used as a source 
  1950                 for the alpha blending. The size of the area is the same as the 
  1951                 source bitmap area - aSrcRect parameter.
  1952 @see CFbsBitGc::AlphaBlendBitmaps()
  1953 */
  1954 	{
  1955 	iPimpl->WriteAnyPendingStateChanges();
  1956 	TWsGcCmdAlphaBlendBitmaps alphaBlend(aDestPt, aSrcBmp->WsHandle(), aSrcRect, aAlphaBmp->WsHandle(), aAlphaPt);
  1957 	Write(&alphaBlend,sizeof(alphaBlend),EWsGcOpGdiWsAlphaBlendBitmaps);
  1958 	AddToBitmapArray(aSrcBmp->Handle());
  1959 	AddToBitmapArray(aAlphaBmp->Handle());
  1960 	return KErrNone;
  1961 	}
  1962 
  1963 /**
  1964 This method has been deprecated. Calling it has no effect.
  1965 @param aDrawOpaque Ignored.
  1966 @deprecated
  1967 */
  1968 EXPORT_C void CWindowGc::SetOpaque(TBool aDrawOpaque)
  1969 	{
  1970 	iPimpl->WriteAnyPendingStateChanges();
  1971 	WriteInt(aDrawOpaque, EWsGcOpSetOpaque);
  1972 	}
  1973 
  1974 /** APIExtension can contain as many additional methods as is required by 
  1975 CGraphicsContext after its original conception. It takes 3 parameters.
  1976 Function is exported due to constrains of retaining BC with earlier versions.
  1977 This is not used directly by external methods, instead it is called by a named 
  1978 method in CGraphicsContext which passes the relivant arguements including an 
  1979 unique identifier for the required action.
  1980 @param aUid The unique identifier for the method that is required. Selected 
  1981 internally by a series of "if" statements. 
  1982 @see Valid Uid identifiers are listed in header gdi.h
  1983 @see CGraphicsContext
  1984 @param aOutput is a TAny pointer to a reference. Used to output data as the structure
  1985 does not need to be instantiated before the function call this adds greater 
  1986 flexibility.
  1987 @param aInput is a TAny pointer used to input data.
  1988 */	
  1989 EXPORT_C TInt CWindowGc::APIExtension(TUid aUid, TAny*& aOutput, TAny* aInput)
  1990 	{
  1991 	if (aUid == KGetUnderlineMetrics)
  1992 		{		
  1993 		return APIExGetUnderlineMetrics(aOutput);
  1994 		}
  1995 	else if (aUid == KSetShadowColor)
  1996 		{
  1997 		return APIExSetShadowColor(aInput);
  1998 		}
  1999 	else if (aUid == KGetShadowColor)
  2000 		{
  2001 		return APIExGetShadowColor(aOutput);
  2002 		}
  2003 	else if (aUid == KDrawTextInContextUid)
  2004 		{
  2005 		TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
  2006 		return APIExDrawText(contextParam->iText, &contextParam->iParam, contextParam->iPosition);
  2007 		}
  2008 	else if (aUid == KDrawBoxTextInContextUid)
  2009 		{
  2010 		TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
  2011 		return APIExDrawText(contextParam->iText,&contextParam->iParam,contextParam->iBox,contextParam->iBaselineOffset,contextParam->iAlign,contextParam->iMargin);
  2012 		}
  2013 	else if (aUid == KDrawTextInContextVerticalUid)
  2014 		{
  2015 		TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
  2016 		return APIExDrawTextVertical(contextParam->iText, &contextParam->iParam, contextParam->iPosition,contextParam->iUp);
  2017 		}
  2018 	else if (aUid == KDrawBoxTextInContextVerticalUid)
  2019 		{
  2020 		TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
  2021 		return APIExDrawTextVertical(contextParam->iText,&contextParam->iParam,contextParam->iBox,contextParam->iBaselineOffset,contextParam->iUp,contextParam->iAlign,contextParam->iMargin);
  2022 		}
  2023 	else if (aUid == KApiExtensionInterfaceUid)
  2024 		{
  2025 		return APIExInterface(aOutput, *static_cast<TUid*>(aInput));
  2026 		}
  2027 	/* Future cases may be placed here later.*/
  2028 	else
  2029 		return CBitmapContext::APIExtension(aUid, aOutput, aInput);
  2030 	}
  2031 
  2032 //The methods listed above in APIExtension follow here with the prefix APIEx.
  2033 TInt CWindowGc::APIExGetUnderlineMetrics(TAny*& aOutput)
  2034 	{
  2035 	const TInt width = Max(iPimpl->iFont->HeightInPixels() / 10,1);
  2036 	TTwoTInt* ptr = (TTwoTInt*)aOutput;
  2037 	ptr->iTop = 1 + width / 2;
  2038 	ptr->iBottom = (ptr->iTop) + width;
  2039 	return KErrNone;
  2040 	}
  2041 
  2042 TInt CWindowGc::APIExSetShadowColor(TAny* aShadowColor)
  2043 	{
  2044 	const TRgb shadowColor = *(reinterpret_cast<TRgb*> (aShadowColor));
  2045 	WriteInt(shadowColor.Internal(), EWsGcOpSetShadowColor);
  2046 	iPimpl->iShadowColor = shadowColor;
  2047 	return KErrNone;
  2048 	}
  2049 
  2050 TInt CWindowGc::APIExGetShadowColor(TAny*& aOutput)
  2051 	{
  2052 	TRgb* ptr = (TRgb*)aOutput;
  2053 	ptr->SetInternal(iPimpl->iShadowColor.Internal());
  2054 	return KErrNone;
  2055 	}
  2056 
  2057 //Default implementation of reserved virtual
  2058 EXPORT_C void CWindowGc::Reserved_CGraphicsContext_2()
  2059 	{
  2060 	CBitmapContext::Reserved_CGraphicsContext_2();
  2061 	}
  2062 
  2063 //Default implementation of reserved virtual
  2064 EXPORT_C void CWindowGc::Reserved_CBitmapContext_1()
  2065 	{
  2066 	CBitmapContext::Reserved_CBitmapContext_1();
  2067 	}
  2068 
  2069 //Default implementation of reserved virtual
  2070 EXPORT_C void CWindowGc::Reserved_CBitmapContext_2()
  2071 	{
  2072 	CBitmapContext::Reserved_CBitmapContext_2();
  2073 	}
  2074 
  2075 //Default implementation of reserved virtual
  2076 EXPORT_C void CWindowGc::Reserved_CBitmapContext_3()
  2077 	{
  2078 	CBitmapContext::Reserved_CBitmapContext_3();
  2079 	}
  2080 	
  2081 // was Reserved_CWindowGc_1
  2082 EXPORT_C void CWindowGc::DrawWsGraphic(const TWsGraphicId& aId,const TRect& aDestRect)
  2083 /** Draws an abstract artwork.
  2084 It does nothing if aDestRect values fall outside the window area.
  2085 
  2086 @param aId the identifier for the artwork
  2087 @param aDestRect the destination rect within the active window for this artwork
  2088 
  2089 @since 9.2
  2090 @released
  2091 */
  2092 	{
  2093 	iPimpl->WriteAnyPendingStateChanges();
  2094 	TWsGcCmdDrawWsGraphic drawWsGraphic(aId,aDestRect);
  2095 	Write(&drawWsGraphic,sizeof(drawWsGraphic),EWsGcOpDrawWsGraphic);
  2096 	}
  2097 
  2098 // Reserved_CWindowGc_2
  2099 EXPORT_C void CWindowGc::DrawWsGraphic(const TWsGraphicId& aId,const TRect& aDestRect,const TDesC8& aData)
  2100 /** Draws an abstract artwork.
  2101 It does nothing if aDestRect values fall outside the window area.
  2102 
  2103 @param aId the identifier for the artwork
  2104 @param aDestRect the destination rect within the active window for this artwork
  2105 @param aData opaque datagram to associate with this occasion of drawing.  The format is dependent upon the artwork
  2106 
  2107 @since 9.2
  2108 @released
  2109 */
  2110 	{
  2111 	iPimpl->WriteAnyPendingStateChanges();
  2112 	TWsGcCmdDrawWsGraphic drawWsGraphic(aId,aDestRect);
  2113 	drawWsGraphic.iDataLen = aData.Size();
  2114 	WriteTextCommand(&drawWsGraphic, sizeof(drawWsGraphic),	aData, EWsGcOpDrawWsGraphic, EWsGcOpDrawWsGraphicPtr);	
  2115 	}
  2116 
  2117 /**
  2118 Gets an extension interface specified by the supplied UID, or NULL if it isn't supported.
  2119 
  2120 @param aInterfaceId The UID of the requested interface
  2121 @return A pointer to the interface, or NULL if the interface isn't supported
  2122 @publishedPartner
  2123 @prototype
  2124 */
  2125 EXPORT_C TAny* CWindowGc::Interface(TUid aInterfaceId)
  2126 	{
  2127 	TAny* interface = NULL;
  2128 	if(KErrNone == APIExtension(KApiExtensionInterfaceUid, interface, &aInterfaceId))
  2129 		return interface;
  2130 	return NULL;
  2131 	}
  2132 
  2133 /**
  2134 Gets an extension interface specified by the supplied UID, or NULL if it isn't supported.
  2135 
  2136 @param aInterfaceId The UID of the requested interface
  2137 @return A pointer to the interface, or NULL if the interface isn't supported
  2138 @publishedPartner
  2139 @prototype
  2140 */
  2141 EXPORT_C const TAny* CWindowGc::Interface(TUid aInterfaceId) const
  2142 	{
  2143 	return const_cast<CWindowGc*>(this)->Interface(aInterfaceId);
  2144 	}
  2145 
  2146 TInt CWindowGc::APIExInterface(TAny*& aInterface, TUid aInterfaceId)
  2147 	{
  2148 	if(aInterfaceId == KMWsDrawResourceInterfaceUid)
  2149 		{
  2150 		aInterface = static_cast<MWsDrawResource*>(iPimpl);
  2151 		return KErrNone;
  2152 		}
  2153 	return KErrNotSupported;
  2154 	}
  2155 
  2156 void CWindowGc::DrawResource(const TPoint& aPos, const RWsDrawableSource& aSource, TGraphicsRotation aRotation)
  2157 	{
  2158 	iPimpl->WriteAnyPendingStateChanges();
  2159 	TWsGcCmdDrawResourceToPos drawWsResource(aSource.WsHandle(), aPos, aRotation);
  2160 	Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceToPos);
  2161 	}
  2162 
  2163 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, TGraphicsRotation aRotation)
  2164 	{
  2165 	iPimpl->WriteAnyPendingStateChanges();
  2166 	TWsGcCmdDrawResourceToRect drawWsResource(aSource.WsHandle(), aDestRect, aRotation);
  2167 	Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceToRect);
  2168 	}
  2169 
  2170 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TRect& aSrcRect, TGraphicsRotation aRotation)
  2171 	{
  2172 	iPimpl->WriteAnyPendingStateChanges();
  2173 	TWsGcCmdDrawResourceFromRectToRect drawWsResource(aSource.WsHandle(), aDestRect, aSrcRect, aRotation);
  2174 	Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceFromRectToRect);
  2175 	}
  2176 
  2177 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TDesC8& aParam)
  2178 	{
  2179 	iPimpl->WriteAnyPendingStateChanges();
  2180 	TWsGcCmdDrawResourceWithData drawWsResource(aSource.WsHandle(), aDestRect, &aParam);
  2181 	Write(&drawWsResource, sizeof(drawWsResource),EWsGcOpDrawResourceWithData);
  2182 	}
  2183 
  2184 //Default implementation of reserved virtual
  2185 EXPORT_C void CWindowGc::Reserved_CWindowGc_3()
  2186 	{
  2187 	}
  2188 
  2189 //Default implementation of reserved virtual
  2190 EXPORT_C void CWindowGc::Reserved_CWindowGc_4()
  2191 	{
  2192 	}
  2193 
  2194 //Default implementation of reserved virtual
  2195 EXPORT_C void CWindowGc::Reserved_CWindowGc_5()
  2196 	{
  2197 	}
  2198 
  2199 /**
  2200 Default constructor. 
  2201 Only for embedding instances of RWsDrawableSource into other classes as data members. 
  2202 Before a RWsDrawableSource can be used the other constructor must be called. 
  2203  */
  2204 EXPORT_C RWsDrawableSource::RWsDrawableSource()
  2205 	: iDrawableId(KSgNullDrawableId), iScreenNumber(KSgScreenIdMain)
  2206 	{
  2207 	}
  2208 
  2209 /**
  2210 Constructor.
  2211 @param aWs Session to the window server
  2212  
  2213 @pre Connection to the window server is established
  2214  */
  2215 EXPORT_C RWsDrawableSource::RWsDrawableSource(RWsSession &aWs)
  2216 	: MWsClientClass(aWs.iBuffer), iDrawableId(KSgNullDrawableId), iScreenNumber(KSgScreenIdMain)
  2217 	{
  2218 	}
  2219 
  2220 /**
  2221 Create window server object for resource drawing operation via window server.
  2222 
  2223 This object will be identified by a unique handle and will be associated with drawable resource which is passed as a parameter.
  2224 
  2225 This object will be created for drawing onto the default screen only.
  2226  
  2227 @see CWindowGc
  2228 @param  aDrawable Drawable resource.
  2229  
  2230 @post Drawable source is created and can be used by window server. The reference counter of the underlying
  2231 		image resource is incremented. 
  2232 
  2233 @return KErrNone if successful, KErrArgument if the image resource is not valid,
  2234 		KErrAlreadyExists if this handle is already associated with a 
  2235 		specific resource,  otherwise one of the system-wide error codes.
  2236  */
  2237 EXPORT_C TInt RWsDrawableSource::Create(const RSgDrawable& aDrawable)
  2238 	{
  2239 	return Create(aDrawable, KSgScreenIdMain);
  2240 	}
  2241 
  2242 /**
  2243 Create window server object for resource drawing operation via window server.
  2244 
  2245 This object will be identified by  unique handle and will be associated with drawable resource which is passed as a parameter.
  2246 
  2247 This object will be created for drawing onto the specified screen only.
  2248 
  2249 @see CWindowGc
  2250 @param  aDrawable Drawable resource.
  2251 @param  aScreenNumber The screen onto which this drawable resource can be drawn.
  2252  
  2253 @post Drawable source is created and can be used by window server. The reference counter of the underlying
  2254 		image resource is incremented. 
  2255 
  2256 @return KErrNone if successful, KErrArgument if the image resource is not valid
  2257 		or if the specified screen is invalid, KErrAlreadyExists if this handle
  2258 		is already associated with a specific resource, otherwise one of the
  2259 		system-wide error codes.
  2260  */
  2261 EXPORT_C TInt RWsDrawableSource::Create(const RSgDrawable& aDrawable, TInt aScreenNumber)
  2262 	{
  2263 	if (iWsHandle)
  2264 		{
  2265 		return KErrAlreadyExists;
  2266 		}
  2267 	CGraphicsResourceWrapperFactory* grwFactory = new CGraphicsResourceWrapperFactory();
  2268 	if (!grwFactory)
  2269 		return KErrNoMemory;
  2270 	// coverity[uninit_use_in_call]
  2271 	CGraphicsResourceWrapper* graphicsResource = grwFactory->NewGraphicsResourceWrapper();
  2272 	if(!graphicsResource)
  2273 		{
  2274 		delete grwFactory;
  2275 		return KErrNotSupported;
  2276 		}
  2277 	if (graphicsResource->IsNull(aDrawable) || aScreenNumber < 0)
  2278 		{
  2279 		delete graphicsResource;
  2280 		delete grwFactory;
  2281 		return KErrArgument;
  2282 		}
  2283 	TWsClCmdCreateDrawableSource create(graphicsResource->Id(aDrawable), aScreenNumber);
  2284 	TInt ret;
  2285 	if ((ret = iBuffer->WriteReplyWs(&create, sizeof(TWsClCmdCreateDrawableSource), EWsClOpCreateDrawableSource)) < 0)
  2286 		{
  2287 		delete graphicsResource;
  2288 		delete grwFactory;
  2289 		return ret;
  2290 		}
  2291 	iWsHandle = ret;
  2292 	iDrawableId = graphicsResource->Id(aDrawable);
  2293 	iScreenNumber = aScreenNumber;
  2294 	delete graphicsResource;
  2295 	delete grwFactory;
  2296 	return KErrNone;
  2297 	}
  2298 
  2299 /**
  2300 Destroy the window server drawable source. 
  2301 Calling this method on a object that is not associated with any RSgDrawable 
  2302 resource will have no effect. Once Close() is called, this drawable source object can be reused.
  2303 
  2304 @post	The window server drawable object is destroyed. The instance is no longer associated
  2305 		with a RSgDrawable specific resource. The reference counter of the underlying
  2306 		image resource is decremented. 
  2307 */
  2308 EXPORT_C void RWsDrawableSource::Close()
  2309 	{
  2310 	if (iWsHandle)
  2311 		{
  2312 		Write(EWsDrawableSourceOpFree);
  2313 		iWsHandle = 0;
  2314 		iDrawableId = KSgNullDrawableId;
  2315 		iScreenNumber = KSgScreenIdMain;
  2316 		}
  2317 	}
  2318 
  2319 /**
  2320  Get the unique ID of the associated drawable resource.
  2321  */
  2322 EXPORT_C const TSgDrawableId& RWsDrawableSource::DrawableId() const
  2323 	{
  2324 	return iDrawableId;
  2325 	}
  2326 
  2327 /**
  2328  Get the screen number of the drawable source.
  2329  */
  2330 EXPORT_C TInt RWsDrawableSource::ScreenNumber() const
  2331 	{
  2332 	return iScreenNumber;
  2333 	}