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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Shells for window server graphics class
19 #include <graphics/wsdrawresource.h>
20 #include "../SERVER/w32cmd.h"
23 #include "graphicsresourcewrapper.h"
24 #include <graphics/gdi/gdiconsts.h>
25 #include <graphics/gdi/gdistructs.h>
26 #include "graphics/windowserverconstants.h"
28 #define KDefaultShadowColor KRgbGray
29 enum {EPolygonMaxHeaderSize=sizeof(TWsCmdHeader)+sizeof(TWsGcCmdSegmentedDrawPolygonData)};
32 // class CWindowGc::CPimpl
35 NONSHARABLE_STRUCT(CWindowGc::CPimpl): public CBase, public MWsDrawResource
36 /** @internalComponent @released */
41 EStateBrushColor = 1<<0,
42 EStateBrushStyle = 1<<1,
43 EStatePenColor = 1<<2,
44 EStatePenStyle = 1<<3,
46 EStateDrawMode = 1<<5,
47 EStateClippingRect = 1<<6,
48 EStateUnderlineStyle = 1<<7,
49 EStateStrikethroughStyle = 1<<8,
50 EStateWordJustification = 1<<9,
51 EStateCharJustification = 1<<10,
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();
61 public: //from MWsDrawResource
62 void DrawResource(const TPoint& aPos, const RWsDrawableSource& aSource, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
64 iGc.DrawResource(aPos, aSource, aRotation);
66 void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
68 iGc.DrawResource(aDestRect, aSource, aRotation);
70 void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TRect& aSrcRect, CWindowGc::TGraphicsRotation aRotation=CWindowGc::EGraphicsRotationNone)
72 iGc.DrawResource(aDestRect, aSource, aSrcRect, aRotation);
74 void DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TDesC8& aParam)
76 iGc.DrawResource(aDestRect, aSource, aParam);
83 TBool iClippingRectSet;
86 TUint32 iPendingState;
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;
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;
114 CWindowGc::CPimpl::CPimpl(CWindowGc& aGc)
115 : iGc(aGc), iFont(NULL), iShadowColor(KDefaultShadowColor)
120 CWindowGc::CPimpl::~CPimpl()
125 void CWindowGc::CPimpl::CancelPendingClippingRect()
127 iPendingState &= ~EStateClippingRect;
130 void CWindowGc::CPimpl::WriteAnyPendingStateChanges()
132 if (iPendingState == 0)
135 if (iPendingState & EStateDrawMode)
137 if (iPendingDrawMode != iCurrentDrawMode)
139 iGc.WriteInt(iPendingDrawMode,EWsGcOpSetDrawMode);
140 iCurrentDrawMode = iPendingDrawMode;
144 if (iPendingState & EStateBrushStyle)
146 if (iPendingBrushStyle != iCurrentBrushStyle)
148 iGc.WriteInt(iPendingBrushStyle,EWsGcOpSetBrushStyle);
149 iCurrentBrushStyle = iPendingBrushStyle;
153 if (iPendingState & EStateBrushColor)
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)
158 iGc.WriteInt(iPendingBrushColor.Internal(),EWsGcOpSetBrushColor);
159 iCurrentBrushColor = iPendingBrushColor;
160 iForceWrite = EFalse;
164 if (iPendingState & EStatePenStyle)
166 if (iPendingPenStyle != iCurrentPenStyle)
168 iGc.WriteInt(iPendingPenStyle,EWsGcOpSetPenStyle);
169 iCurrentPenStyle = iPendingPenStyle;
173 if (iPendingState & EStatePenColor)
175 if (iPendingPenColor != iCurrentPenColor)
177 iGc.WriteInt(iPendingPenColor.Internal(),EWsGcOpSetPenColor);
178 iCurrentPenColor = iPendingPenColor;
182 if (iPendingState & EStateClippingRect)
184 iGc.WriteRect(iPendingClippingRect,EWsGcOpSetClippingRect);
185 iClippingRectSet = ETrue;
188 if (iPendingState & EStateUnderlineStyle)
190 if (iPendingUnderlineStyle != iCurrentUnderlineStyle)
192 iGc.WriteInt(iPendingUnderlineStyle,EWsGcOpSetUnderlineStyle);
193 iCurrentUnderlineStyle = iPendingUnderlineStyle;
197 if (iPendingState & EStateStrikethroughStyle)
199 if (iPendingStrikethroughStyle != iCurrentStrikethroughStyle)
201 iGc.WriteInt(iPendingStrikethroughStyle,EWsGcOpSetStrikethroughStyle);
202 iCurrentStrikethroughStyle = iPendingStrikethroughStyle;
206 if (iPendingState & EStatePenSize)
208 if (iPendingPenSize != iCurrentPenSize)
210 iGc.WriteSize(iPendingPenSize,EWsGcOpSetPenSize);
211 iCurrentPenSize = iPendingPenSize;
215 if (iPendingState & EStateWordJustification)
217 if (iPendingWordJustification.excessWidth != iCurrentWordJustification.excessWidth
218 || iPendingWordJustification.numGaps != iCurrentWordJustification.numGaps)
220 iGc.Write(&iPendingWordJustification,sizeof(iPendingWordJustification),EWsGcOpSetWordJustification);
221 iCurrentWordJustification = iPendingWordJustification;
225 if (iPendingState & EStateCharJustification)
227 if (iPendingCharJustification.excessWidth != iCurrentCharJustification.excessWidth
228 || iPendingCharJustification.numGaps != iCurrentCharJustification.numGaps)
230 iGc.Write(&iPendingCharJustification,sizeof(iPendingCharJustification),EWsGcOpSetCharJustification);
231 iCurrentCharJustification = iPendingCharJustification;
238 void CWindowGc::CPimpl::ResetPendingState()
240 // This function should only be called from CWindowGc::Reset()
241 iForceWrite = EFalse;
244 iClippingRectSet = EFalse;
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;
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;
274 void CWindowGc::CPimpl::StorePendingStateChange(TStateType aState, const TAny* aValue)
278 case EStateBrushColor:
279 iPendingState |= EStateBrushColor;
280 iPendingBrushColor = *((TRgb*)(aValue));
282 case EStateBrushStyle:
283 iPendingState|=EStateBrushStyle;
284 iPendingBrushStyle = *((TBrushStyle*)(aValue));
287 iPendingState|=EStatePenColor;
288 iPendingPenColor = *((TRgb*)(aValue));
291 iPendingState|=EStatePenStyle;
292 iPendingPenStyle = *((TPenStyle*)(aValue));
295 iPendingState|=EStateDrawMode;
296 iPendingDrawMode = *((TDrawMode*)(aValue));
298 case EStateClippingRect:
299 iPendingState|=EStateClippingRect;
300 iPendingClippingRect = *((TRect*)(aValue));
302 case EStateUnderlineStyle:
303 iPendingState|=EStateUnderlineStyle;
304 iPendingUnderlineStyle = *((TFontUnderline*)(aValue));
306 case EStateStrikethroughStyle:
307 iPendingState|=EStateStrikethroughStyle;
308 iPendingStrikethroughStyle= *((TFontStrikethrough*)(aValue));
311 iPendingState|=EStatePenSize;
312 iPendingPenSize = *((TSize*)(aValue));
314 case EStateWordJustification:
315 iPendingState|=EStateWordJustification;
316 iPendingWordJustification = *((TWsGcCmdSetJustification*)(aValue));
318 case EStateCharJustification:
319 iPendingState|=EStateCharJustification;
320 iPendingCharJustification = *((TWsGcCmdSetJustification*)(aValue));
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.
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() */
339 EXPORT_C CWindowGc::~CWindowGc()
342 if (iBuffer && iWsHandle)
347 void CWindowGc::WriteTextCommand(TAny *cmd, TInt len,const TDesC &aBuf,TInt opcode,TInt opcodePtr) const
349 if ((aBuf.Size()+len)>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)))
351 WriteReplyByProvidingRemoteReadAccess(cmd,len,&aBuf,opcodePtr);
355 Write(cmd,len,aBuf.Ptr(),aBuf.Size(),opcode);
359 void CWindowGc::WriteTextCommand(TAny *cmd, TInt len,const TDesC8 &aBuf,TInt opcode,TInt opcodePtr) const
361 if ((aBuf.Size()+len)>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)))
363 WriteReplyByProvidingRemoteReadAccess(cmd,len,&aBuf,opcodePtr);
367 Write(cmd,len,aBuf.Ptr(),aBuf.Size(),opcode);
371 EXPORT_C TInt CWindowGc::Construct()
372 /** Completes construction.
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. */
378 __ASSERT_DEBUG(iWsHandle == KNullHandle, Panic(EW32PanicGraphicDoubleConstruction));
379 iPimpl = new CPimpl(*this);
383 if ((ret=iBuffer->WriteReplyWs(EWsClOpCreateGc))<0)
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.
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().
398 @param aWindow The window for which the graphics context is to be activated. */
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
405 EXPORT_C void CWindowGc::Deactivate()
406 /** Frees the graphics context to be used with another window.
408 This method should be called when the application has completed drawing to
411 iPimpl->ResetPendingState(); // needed because server-side state is reset on deactivation
412 Write(EWsGcOpDeactivate);
414 iPimpl->iShadowColor = KDefaultShadowColor;
417 //====================Functions from GDI.H===============================
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.
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*/
430 EXPORT_C void CWindowGc::SetOrigin(const TPoint &aPoint)
431 /** Sets the position of the co-ordinate origin.
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.
436 @param aPoint A point for the origin, default (0,0).
437 @see CGraphicsContext::SetOrigin() */
439 if ( iPimpl->GetState() & CPimpl::EStateClippingRect )
441 iPimpl->WriteAnyPendingStateChanges();
443 WritePoint(aPoint,EWsGcOpSetOrigin);
446 EXPORT_C void CWindowGc::SetClippingRect(const TRect& aRect)
447 /** Sets a clipping rectangle.
449 Graphics drawn to the window are clipped, so that only items which fall within
450 the rectangle are displayed.
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.
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().
459 @see SetClippingRegion() */
462 iPimpl->StorePendingStateChange(CPimpl::EStateClippingRect, &rect);
465 EXPORT_C void CWindowGc::CancelClippingRect()
466 /** Cancels the clipping rectangle.
468 @see SetClippingRect() */
470 if (iPimpl->GetState() & CPimpl::EStateClippingRect)
472 iPimpl->CancelPendingClippingRect();
475 if (iPimpl->iClippingRectSet)
477 Write(EWsGcOpCancelClippingRect);
480 iPimpl->iClippingRectSet = EFalse;
483 EXPORT_C TInt CWindowGc::SetClippingRegion(const TRegion &aRegion)
484 /** Sets the clipping region.
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.
489 This function always causes a flush of the window server buffer.
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.
495 @return KErrNone if successful, KErrNoMemory if there is insufficient memory
496 to create the region on the server side, otherwise another error code.
498 @see SetClippingRect() */
500 const TInt regionCount=aRegion.Count();
501 TPtrC8 ptrRect(reinterpret_cast<const TUint8*>(aRegion.RectangleList()),regionCount*sizeof(TRect));
502 return(WriteReplyByProvidingRemoteReadAccess(®ionCount,sizeof(regionCount),&ptrRect,EWsGcOpSetClippingRegion));
505 EXPORT_C void CWindowGc::CancelClippingRegion()
506 /** Cancels the current clipping region. */
508 Write(EWsGcOpCancelClippingRegion);
511 EXPORT_C void CWindowGc::SetDrawMode(TDrawMode aDrawingMode)
512 /** Sets the drawing mode.
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
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).
523 The three most important modes are TDrawMode::EDrawModePEN, TDrawMode::EDrawModeNOTSCREEN
524 and TDrawMode::EDrawModeXOR. The default drawing mode is TDrawMode::EDrawModePEN.
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).
532 TDrawMode::EDrawModeAND gives a "colour filter" effect. For example:
534 ANDing with white gives the original colour
536 ANDing with black gives black
538 TDrawMode::EDrawModeOR gives a "colour boost" effect. For example:
540 ORing with black gives the original colour
542 ORing with white gives white
544 TDrawMode::EDrawModeXOR gives an "Exclusive OR" effect. For example:
546 white XOR black gives white
548 white XOR white gives black
550 black XOR black gives black
552 @param aDrawingMode A drawing mode.
553 @see CGraphicsContext::SetDrawMode() */
555 iPimpl->StorePendingStateChange(CPimpl::EStateDrawMode, &aDrawingMode);
558 EXPORT_C void CWindowGc::UseFont(const CFont *aFont)
559 /** Sets this context's font.
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.
564 Note that this function must be called prior to drawing text or the calling
567 @param aFont A device font.
568 @see CGraphicsContext::UseFont() */
570 if (iPimpl->iFont!=(CFbsFont *)aFont)
572 iPimpl->iFont=(CFbsFont *)aFont;
573 WriteInt(iPimpl->iFont->Handle(),EWsGcOpUseFont);
577 EXPORT_C void CWindowGc::DiscardFont()
580 This frees up the memory used (if the font is not being shared with some other
583 Note that if no font is in use when this function is called, then there is no effect.
585 @see CGraphicsContext::DiscardFont() */
587 Write(EWsGcOpDiscardFont);
591 EXPORT_C void CWindowGc::SetUnderlineStyle(TFontUnderline aUnderlineStyle)
592 /** Sets the underline style for all subsequently drawn text.
594 @param aUnderlineStyle The underline style: either on or off.
595 @see CGraphicsContext::SetUnderlineStyle() */
597 iPimpl->StorePendingStateChange(CPimpl::EStateUnderlineStyle, &aUnderlineStyle);
600 EXPORT_C void CWindowGc::SetStrikethroughStyle(TFontStrikethrough aStrikethroughStyle)
601 /** Sets the strikethrough style for all subsequently drawn text.
603 @param aStrikethroughStyle The strikethrough style: either on or off.
604 @see CGraphicsContext::SetStrikethroughStyle() */
606 iPimpl->StorePendingStateChange(CPimpl::EStateStrikethroughStyle, &aStrikethroughStyle);
609 void CWindowGc::SetJustification(TInt aExcessWidth,TInt aNumGaps, TInt aOpcode)
611 TWsGcCmdSetJustification justification;
613 justification.excessWidth=aExcessWidth;
614 justification.numGaps=aNumGaps;
616 if (aOpcode == EWsGcOpSetWordJustification)
618 iPimpl->StorePendingStateChange(CPimpl::EStateWordJustification, &justification);
620 else if (aOpcode == EWsGcOpSetCharJustification)
622 iPimpl->StorePendingStateChange(CPimpl::EStateCharJustification, &justification);
626 EXPORT_C void CWindowGc::SetWordJustification(TInt aExcessWidth,TInt aNumGaps)
627 /** Sets word justification.
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.
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() */
638 SetJustification(aExcessWidth, aNumGaps, EWsGcOpSetWordJustification);
641 EXPORT_C void CWindowGc::SetCharJustification(TInt aExcessWidth,TInt aNumChars)
642 /** Sets the character justification.
644 This function is used primarily to get accurate WYSIWYG, and is not intended
645 for regular use by developers.
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
654 This function is particularly useful for WYSIWYG underlining or strikethrough,
655 as it ensures that the lines extend into the gaps between characters.
657 See CGraphicsContext::SetCharJustification() for more information.
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() */
664 SetJustification(aExcessWidth, aNumChars, EWsGcOpSetCharJustification);
667 EXPORT_C void CWindowGc::SetPenColor(const TRgb &aColor)
668 /** Sets the pen colour.
670 The effective pen colour depends on the drawing mode (see SetDrawMode()).
672 The default pen colour is black.
674 @param aColor The RGB colour for the pen.
675 @see CGraphicsContext::SetPenColor() */
678 iPimpl->StorePendingStateChange(CPimpl::EStatePenColor, &color);
681 EXPORT_C void CWindowGc::SetPenStyle(TPenStyle aPenStyle)
682 /** Sets the line drawing style for the pen.
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.
688 To use a pen style, its full context must be given, e.g. for a null pen: CGraphicsContext::TPenStyle::ENullPen.
690 @param aPenStyle A pen style.
691 @see CGraphicsContext::SetPenStyle() */
693 iPimpl->StorePendingStateChange(CPimpl::EStatePenStyle, &aPenStyle);
696 EXPORT_C void CWindowGc::SetPenSize(const TSize& aSize)
697 /** Sets the line drawing size for the pen.
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
705 @param aSize A line size, the default being 1 pixel.
706 @see CGraphicsContext::SetPenSize() */
709 iPimpl->StorePendingStateChange(CPimpl::EStatePenSize, &size);
712 EXPORT_C void CWindowGc::SetBrushColor(const TRgb &aColor)
713 /** Sets the brush colour.
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.
720 @param aColor The RGB colour for the brush.
721 @see CGraphicsContext::SetBrushColor() */
724 iPimpl->StorePendingStateChange(CPimpl::EStateBrushColor, &color);
727 EXPORT_C void CWindowGc::SetBrushStyle(TBrushStyle aBrushStyle)
728 /** Sets the line drawing style for the brush.
730 The GDI provides ten brush styles, including six built-in hatching patterns
731 (see CGraphicsContext::TBrushStyle).
733 Use TBrushStyle::ENullBrush to draw the outline of a fillable shape on its
734 own, without filling.
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.
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().
743 @param aBrushStyle The brush style.
744 @see CGraphicsContext::SetBrushStyle() */
746 iPimpl->StorePendingStateChange(CPimpl::EStateBrushStyle, &aBrushStyle);
748 EXPORT_C void CWindowGc::SetBrushOrigin(const TPoint &aOrigin)
749 /** Sets the brush pattern origin.
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.
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.
762 If SetBrushOrigin() is not used, then the origin defaults to (0,0).
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).
769 @param aOrigin The origin point for the brush.
770 @see CGraphicsContext::SetBrushOrigin() */
772 WritePoint(aOrigin,EWsGcOpSetBrushOrigin);
775 EXPORT_C void CWindowGc::UseBrushPattern(const CFbsBitmap *aDevice)
776 /** Sets the brush pattern to the specified bitmap.
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.
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.
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().
792 @param aDevice A bitmap pattern for the brush
793 @see CGraphicsContext::UseBrushPattern() */
795 WriteInt(aDevice->Handle(),EWsGcOpUseBrushPattern);
796 AddToBitmapArray(aDevice->Handle());
799 EXPORT_C void CWindowGc::DiscardBrushPattern()
800 /** Discards a non-built-in brush pattern.
802 This frees up the memory used for the bitmap, if it is not being shared by
805 If no brush pattern has been set when this function is called, it has no effect.
807 @see CGraphicsContext::DiscardBrushPattern() */
809 Write(EWsGcOpDiscardBrushPattern);
812 EXPORT_C void CWindowGc::Plot(const TPoint &aPoint)
813 /** Draws a single point.
815 The point is drawn with the current pen settings using the current drawing
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().
823 @param aPoint The point to be drawn.
824 @see CGraphicsContext::Plot() */
826 iPimpl->WriteAnyPendingStateChanges();
827 WritePoint(aPoint,EWsGcOpPlot);
830 EXPORT_C void CWindowGc::DrawLine(const TPoint &aPoint1,const TPoint &aPoint2)
831 /** Draws a straight line between two points.
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() */
837 iPimpl->WriteAnyPendingStateChanges();
838 TWsGcCmdDrawLine drawLine(aPoint1,aPoint2);
839 Write(&drawLine,sizeof(drawLine),EWsGcOpDrawLine);
842 EXPORT_C void CWindowGc::MoveTo(const TPoint &aPoint)
843 /** Moves the internal drawing position relative to the co-ordinate origin, without
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.
851 The operations DrawLine(), DrawLineTo(), DrawLineBy() and DrawPolyline() also
852 change the internal drawing position to the last point of the drawn line(s).
854 The internal drawing position is set to the co-ordinate origin if no drawing
855 or moving operations have yet taken place.
857 @param aPoint The point to move the internal drawing position to.
858 @see CGraphicsContext::MoveTo()
859 @see CGraphicsContext::MoveBy() */
861 WritePoint(aPoint,EWsGcOpMoveTo);
864 EXPORT_C void CWindowGc::MoveBy(const TPoint &aPoint)
865 /** Moves the internal drawing position by a vector, without drawing a line.
867 The internal drawing position is moved relative to its current co-ordinates.
869 @param aPoint The vector to move the internal drawing position by.
870 @see CGraphicsContext::MoveBy()
871 @see CGraphicsContext::MoveTo() */
873 WritePoint(aPoint,EWsGcOpMoveBy);
876 EXPORT_C void CWindowGc::DrawLineTo(const TPoint &aPoint)
877 /** Draws a straight line from the current internal drawing position to a point.
879 @param aPoint The point at the end of the line.
880 @see CGraphicsContext::DrawLineTo() */
882 iPimpl->WriteAnyPendingStateChanges();
883 WritePoint(aPoint,EWsGcOpDrawTo);
886 EXPORT_C void CWindowGc::DrawLineBy(const TPoint &aPoint)
887 /** Draws a straight line relative to the current internal drawing position, using
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
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() */
898 iPimpl->WriteAnyPendingStateChanges();
899 WritePoint(aPoint,EWsGcOpDrawBy);
902 void CWindowGc::doDrawPolyLine(const CArrayFix<TPoint> *aPointArray, const TPoint* aPointList,TInt aNumPoints)
904 TWsGcOpcodes opcode=EWsGcOpDrawPolyLine;
905 TWsGcCmdDrawPolyLine polyLine;
906 TInt maxBufLen=(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(polyLine))/sizeof(TPoint);
908 while(sent<aNumPoints)
914 ptr=&(*aPointArray)[sent];
915 availableLen=aPointArray->End(sent)-ptr;
920 availableLen=aNumPoints-sent;
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;
931 EXPORT_C void CWindowGc::DrawPolyLine(const TPoint* aPointList,TInt aNumPoints)
932 /** Draws a polyline using points in a list.
934 A polyline is a series of concatenated straight lines joining a set of points.
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() */
940 iPimpl->WriteAnyPendingStateChanges();
941 doDrawPolyLine(NULL,aPointList,aNumPoints);
944 EXPORT_C void CWindowGc::DrawPolyLine(const CArrayFix<TPoint> *aPointArray)
945 /** Draws a polyline using points in an array.
947 A polyline is a series of concatenated straight lines joining a set of points.
949 @param aPointArray An array containing the points on the polyline.
950 @see CGraphicsContext::DrawPolyLine() */
952 iPimpl->WriteAnyPendingStateChanges();
953 doDrawPolyLine(aPointArray,NULL,aPointArray->Count());
956 TInt CWindowGc::doDrawPolygon(const CArrayFix<TPoint> *aPointArray,const TPoint* aPointList,TInt aNumPoints,TFillRule aFillRule)
960 TWsGcCmdSegmentedDrawPolygonData polyData;
962 TInt maxBufLen=(iBuffer->BufferSize()-EPolygonMaxHeaderSize)/sizeof(TPoint);
969 ptr=&(*aPointArray)[polyData.index];
970 availableLen=aPointArray->End(polyData.index)-ptr;
974 ptr=aPointList+polyData.index;
975 availableLen=aNumPoints-polyData.index;
977 polyData.numPoints=Min(availableLen,maxBufLen);
978 if (polyData.index==0) // First time around
980 if (polyData.numPoints==aNumPoints) // Can it be done in one go?
982 TWsGcCmdDrawPolygon drawPolygon;
983 drawPolygon.numPoints=aNumPoints;
984 drawPolygon.fillRule=aFillRule;
985 Write(&drawPolygon,sizeof(drawPolygon),ptr,aNumPoints*sizeof(TPoint),EWsGcOpDrawPolygon);
988 TWsGcCmdStartSegmentedDrawPolygon start;
989 start.totalNumPoints=aNumPoints;
990 TInt err=WriteReply(&start,sizeof(start),EWsGcOpStartSegmentedDrawPolygon);
994 Write(&polyData,sizeof(polyData),ptr,polyData.numPoints*sizeof(TPoint),EWsGcOpSegmentedDrawPolygonData);
995 polyData.index+=polyData.numPoints;
996 if (polyData.index==aNumPoints)
998 TWsGcCmdDrawSegmentedPolygon draw;
999 draw.fillRule=aFillRule;
1000 Write(&draw,sizeof(draw),EWsGcOpDrawSegmentedPolygon);
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.
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.
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
1025 The filling of a polygon proceeds according to this algorithm:
1027 If aFillRule is TFillRule::EAlternate (default) and it has an odd winding
1028 number, then fill the surrounding area.
1030 If aFillRule is TFillRule::EWinding and it has a winding number greater than
1031 zero, then fill the surrounding area.
1033 This function always causes a flush of the window server buffer.
1035 @param aPointList Pointer to a list of points, specifying the vertices of
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
1041 @see CGraphicsContext::DrawPolygon() */
1043 iPimpl->WriteAnyPendingStateChanges();
1044 return(doDrawPolygon(NULL,aPointList,aNumPoints,aFillRule));
1047 EXPORT_C TInt CWindowGc::DrawPolygon(const CArrayFix<TPoint> *aPointArray,TFillRule aFillRule)
1048 /** Draws and fills a polygon using points defined in an array.
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.
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
1065 The filling of a polygon proceeds according to this algorithm:
1067 If aFillRule is TFillRule::EAlternate (default) and it has an odd winding
1068 number, then fill the surrounding area.
1070 If aFillRule is TFillRule::EWinding and it has a winding number greater than
1071 zero, then fill the surrounding area.
1073 This function always causes a flush of the window server buffer.
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
1079 @see CGraphicsContext::DrawPolygon() */
1081 iPimpl->WriteAnyPendingStateChanges();
1082 return(doDrawPolygon(aPointArray,NULL,aPointArray->Count(),aFillRule));
1085 void CWindowGc::DrawArcOrPie(const TRect &aRect,const TPoint &aStart,const TPoint &aEnd, TInt aOpcode)
1087 iPimpl->WriteAnyPendingStateChanges();
1088 TWsGcCmdDrawArcOrPie cmd(aRect,aStart,aEnd);
1089 Write(&cmd,sizeof(cmd),aOpcode);
1092 EXPORT_C void CWindowGc::DrawArc(const TRect &aRect,const TPoint &aStart,const TPoint &aEnd)
1093 /** Draws an arc (a portion of an ellipse).
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.
1102 The arc itself is the segment of the ellipse in an anti-clockwise direction
1103 from the start point to the end point.
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.
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.
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.
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.
1121 Line drawing is subject to pen colour, width and style and draw mode
1123 @param aRect The rectangle in which to draw the ellipse (of which the arc is
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() */
1129 iPimpl->WriteAnyPendingStateChanges();
1130 DrawArcOrPie(aRect,aStart,aEnd,EWsGcOpDrawArc);
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.
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.
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.
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.
1153 The line drawn by the pen goes inside the rectangle given by the aRect argument.
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.
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
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.
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.
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() */
1179 iPimpl->WriteAnyPendingStateChanges();
1180 DrawArcOrPie(aRect,aStart,aEnd,EWsGcOpDrawPie);
1183 EXPORT_C void CWindowGc::DrawEllipse(const TRect &aRect)
1184 /** Draws and fills an ellipse.
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.
1190 The column and row of pixels containing the bottom right co-ordinate of the
1191 aRect argument are not part of the rectangle.
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.
1198 @param aRect The rectangle in which to draw the ellipse
1199 @see CGraphicsContext::DrawEllipse() */
1201 iPimpl->WriteAnyPendingStateChanges();
1202 WriteRect(aRect,EWsGcOpDrawEllipse);
1205 EXPORT_C void CWindowGc::DrawRect(const TRect &aRect)
1206 /** Draws and fills a rectangle.
1208 The rectangle's border is drawn with the pen, and it is filled using the brush.
1210 @param aRect The rectangle to be drawn.
1211 @see CGraphicsContext::DrawRect() */
1213 iPimpl->WriteAnyPendingStateChanges();
1214 WriteRect(aRect,EWsGcOpDrawRect);
1217 EXPORT_C void CWindowGc::DrawRoundRect(const TRect &aRect,const TSize &aEllipse)
1218 /** Draws and fills a rectangle with rounded corners.
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.
1224 The line drawn by the pen (if any) goes inside the rectangle given by the
1229 Dotted and dashed pen styles cannot be used for the outline of a rounded rectangle.
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.
1234 @param aRect The rectangle to be drawn.
1235 @param aEllipse The dimensions of each corner.
1236 @see CGraphicsContext::DrawRoundRect() */
1238 iPimpl->WriteAnyPendingStateChanges();
1239 TWsGcCmdDrawRoundRect drawRoundRect(aRect,aEllipse);
1240 Write(&drawRoundRect,sizeof(drawRoundRect),EWsGcOpDrawRoundRect);
1243 EXPORT_C void CWindowGc::DrawBitmap(const TPoint &aTopLeft, const CFbsBitmap *aDevice)
1244 /** Draws a bitmap at a specified point.
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).
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.
1256 Note: this member function uses the bitmap's size in twips and does a stretch/compress
1257 blit using a linear DDA.
1259 @param aTopLeft The point where the top left pixel of the bitmap is to be
1261 @param aDevice The source bitmap.
1262 @see CGraphicsContext::DrawBitmap() */
1264 iPimpl->WriteAnyPendingStateChanges();
1265 TWsGcCmdDrawBitmap drawBitmap(aTopLeft,aDevice->Handle());
1266 Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap);
1267 AddToBitmapArray(aDevice->Handle());
1270 EXPORT_C void CWindowGc::DrawBitmap(const TRect &aDestRect, const CFbsBitmap *aDevice)
1271 /** Draws a bitmap in a rectangle.
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).
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.
1283 Notes: this member function uses the bitmap's size in pixels and does a stretch/compress
1284 blit using a linear DDA.
1286 @param aDestRect The rectangle within which the bitmap is to be drawn.
1287 @param aDevice The source bitmap.
1288 @see CGraphicsContext::DrawBitmap() */
1290 iPimpl->WriteAnyPendingStateChanges();
1291 TWsGcCmdDrawBitmap2 drawBitmap(aDestRect,aDevice->Handle());
1292 Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap2);
1293 AddToBitmapArray(aDevice->Handle());
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.
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).
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.
1309 Note: this member function uses rectangle sizes in pixels and does a stretch/compress
1310 blit using a linear DDA.
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() */
1318 iPimpl->WriteAnyPendingStateChanges();
1319 TWsGcCmdDrawBitmap3 drawBitmap(aDestRect,aDevice->Handle(),aSourceRect);
1320 Write(&drawBitmap,sizeof(drawBitmap),EWsGcOpDrawBitmap3);
1321 AddToBitmapArray(aDevice->Handle());
1324 /** Draws a specified rectangle from a bitmap and its mask into another rectangle.
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.
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.
1337 If mask bitmap's display mode is EColor256, the function does AplhaBlending
1338 and ignores aInvertMask parameter.
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.
1346 Note: this member function uses rectangle sizes in pixels and does a stretch/compress
1347 blit using a linear DDA.
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)
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());
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
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.
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.
1381 If mask bitmap's display mode is EColor256, the function does AplhaBlending
1382 and ignores aInvertMask parameter.
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.
1390 Note: this member function uses rectangle sizes in pixels and does a stretch/compress
1391 blit using a linear DDA.
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. */
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());
1408 EXPORT_C void CWindowGc::DrawText(const TDesC &aBuf, const TPoint &aPos)
1409 /** Draws horizontal text with no surrounding box.
1411 The appearance of the text is subject to the drawing mode, the font, pen colour,
1412 word justification and character justification.
1414 A panic occurs if this function is called when there is no font: see UseFont().
1416 @param aBuf The string to write.
1417 @param aPos The point specifying the position of the baseline at the left
1419 @see CGraphicsContext::DrawText() */
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);
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.
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).
1434 A panic occurs if this function is called when there is no font: see UseFont().
1436 Note: the text is clipped to the box. You must ensure that the specified string
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
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() */
1450 iPimpl->WriteAnyPendingStateChanges();
1451 if (aBuf.Size()<(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(TWsGcCmdBoxTextOptimised2)))
1453 if (aHoriz==ELeft && aLeftMrg==0)
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);
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);
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);
1474 TInt CWindowGc::APIExDrawText(const TDesC& aBuf,const TTextParameters* aParam,const TPoint& aPos)
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);
1483 TInt CWindowGc::APIExDrawText(const TDesC& aBuf,const TTextParameters* aParam,const TRect& aBox,TInt aBaselineOffset,TTextAlign aHoriz,TInt aLeftMrg)
1485 iPimpl->WriteAnyPendingStateChanges();
1486 if (aBuf.Size()<(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)-sizeof(TWsGcCmdBoxTextInContextOptimised2)))
1488 if (aHoriz==ELeft && aLeftMrg==0)
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);
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);
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);
1509 EXPORT_C void CWindowGc::DrawTextVertical(const TDesC& aText,const TPoint& aPos,TBool aUp)
1510 /** Draws vertical text in the specified direction.
1512 A panic occurs if this function is called when there is no font: see UseFont().
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. */
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);
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
1528 A panic occurs if this function is called when there is no font: see UseFont().
1530 @param aText The text to be drawn.
1531 @param aBox The bounding box within which the text should be drawn, and which
1533 @param aBaselineOffset The height of the top of the characters from their text
1535 @param aUp The direction. ETrue for up, EFalse for down.
1536 @param aVert The text alignment.
1537 @param aMargin The margin. */
1539 iPimpl->WriteAnyPendingStateChanges();
1540 TWsGcCmdBoxTextVertical boxText(aBox);
1541 boxText.baselineOffset=aBaselineOffset;
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);
1551 TInt CWindowGc::APIExDrawTextVertical(const TDesC& aText,const TTextParameters* aParam,const TPoint& aPos,TBool aUp)
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);
1560 TInt CWindowGc::APIExDrawTextVertical(const TDesC& aText,const TTextParameters* aParam,const TRect& aBox,TInt aBaselineOffset,TBool aUp,TTextAlign aVert,TInt aMargin)
1562 iPimpl->WriteAnyPendingStateChanges();
1563 TWsGcCmdBoxTextInContextVertical boxText(aBox);
1564 boxText.baselineOffset=aBaselineOffset;
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);
1577 //========================Extra functions============================
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
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
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).
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).
1595 This version of this function is only really suitable for testing.
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
1602 @see CBitmapContext::CopyRect() */
1604 iPimpl->WriteAnyPendingStateChanges();
1605 TWsGcCmdCopyRect copyRect(anOffset,aRect);
1606 Write(©Rect,sizeof(copyRect),EWsGcOpCopyRect);
1609 EXPORT_C void CWindowGc::BitBlt(const TPoint &aPoint, const CFbsBitmap *aBitmap)
1610 /** Performs a bitmap block transfer.
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.
1618 @param aPoint The position for the top left corner of the bitmap.
1619 @param aBitmap A memory-resident bitmap.
1620 @see CBitmapContext::BitBlt() */
1622 if (aBitmap == NULL || !aBitmap->Handle())
1624 iPimpl->WriteAnyPendingStateChanges();
1625 TWsGcCmdGdiBlt2 gdiBlit(aPoint,aBitmap->Handle());
1626 Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiBlt2);
1627 AddToBitmapArray(aBitmap->Handle());
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.
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.
1639 Note: if the rectangle aSource is larger than the bitmap then the bitmap will be padded
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() */
1648 if (aBitmap == NULL || !aBitmap->Handle())
1650 iPimpl->WriteAnyPendingStateChanges();
1651 TWsGcCmdGdiBlt3 gdiBlit(aDestination,aBitmap->Handle(),aSource);
1652 Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiBlt3);
1653 AddToBitmapArray(aBitmap->Handle());
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.
1659 The mask bitmap can be used as either a positive or negative mask. Masked
1660 pixels are not mapped to the destination rectangle.
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.
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.
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.
1683 @see CBitmapContext::BitBltMasked() */
1685 if (aBitmap == NULL || !aBitmap->Handle() || aMaskBitmap == NULL || !aMaskBitmap->Handle())
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());
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
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.
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.
1707 @param aPoint The position for the top left corner of the bitmap.
1708 @param aBitmap A window server bitmap.
1709 @see CBitmapContext::BitBlt() */
1711 if (aBitmap == NULL || !aBitmap->Handle())
1713 iPimpl->WriteAnyPendingStateChanges();
1714 TWsGcCmdGdiBlt2 gdiBlit(aPoint,aBitmap->WsHandle());
1715 Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiWsBlt2);
1716 AddToBitmapArray(aBitmap->Handle());
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.
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.
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.
1732 Note: if the rectangle aSource is larger than the bitmap then the bitmap will be padded
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() */
1741 if (aBitmap == NULL || !aBitmap->Handle())
1743 iPimpl->WriteAnyPendingStateChanges();
1744 TWsGcCmdGdiBlt3 gdiBlit(aDestination,aBitmap->WsHandle(),aSource);
1745 Write(&gdiBlit,sizeof(gdiBlit),EWsGcOpGdiWsBlt3);
1746 AddToBitmapArray(aBitmap->Handle());
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.
1752 The mask bitmap can be used as either a positive or negative mask. Masked
1753 pixels are not mapped to the destination rectangle.
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.
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.
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.
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.
1779 @see CBitmapContext::BitBltMasked() */
1781 if (aBitmap == NULL || !aBitmap->Handle() || aMaskBitmap == NULL || !aMaskBitmap->Handle())
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());
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.
1799 EXPORT_C void CWindowGc::MapColors(const TRect& /*aRect*/, const TRgb* /*aColors*/, TInt /*aNumPairs*/, TBool /*aMapForwards*/)
1803 EXPORT_C void CWindowGc::Clear(const TRect &aRect)
1804 /** Clears a rectangular area of a window.
1806 The cleared area is filled with the current brush colour.
1808 @param aRect The rectangle to clear.
1809 @see CBitmapContext::Clear() */
1811 iPimpl->WriteAnyPendingStateChanges();
1812 WriteRect(aRect,EWsGcOpClearRect);
1815 EXPORT_C void CWindowGc::Clear()
1816 /** Clears the whole window.
1818 The cleared area is filled with the current brush colour.
1820 @see CBitmapContext::Clear() */
1822 iPimpl->WriteAnyPendingStateChanges();
1823 Write(EWsGcOpClear);
1826 EXPORT_C void CWindowGc::Reset()
1827 /** Resets the graphics context to its default settings.
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.
1833 @see CGraphicsContext::Reset() */
1835 Write(EWsGcOpReset);
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
1843 This method has been deprecated. Dithering is no longer supported. Calling it
1845 @param aPoint Ignored.
1848 EXPORT_C void CWindowGc::SetDitherOrigin(const TPoint& /*aPoint*/)
1852 EXPORT_C void CWindowGc::SetFaded(TBool aFaded)
1853 /** Sets whether the graphics context is faded.
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
1859 @param aFaded ETrue to fade the graphics context, EFalse to unfade it. */
1861 iPimpl->WriteAnyPendingStateChanges();
1862 WriteInt(aFaded,EWsGcOpSetFaded);
1865 EXPORT_C void CWindowGc::SetFadingParameters(TUint8 aBlackMap,TUint8 aWhiteMap)
1866 /** Sets the fading parameters.
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.
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
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().
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).
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() */
1892 WriteInt(WservEncoding::Encode8BitValues(aBlackMap,aWhiteMap),EWsGcOpSetFadeParams);
1895 EXPORT_C TInt CWindowGc::AlphaBlendBitmaps(const TPoint& aDestPt, const CFbsBitmap* aSrcBmp, const TRect& aSrcRect,const CFbsBitmap* aAlphaBmp, const TPoint& aAlphaPt)
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.
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.
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()
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());
1932 EXPORT_C TInt CWindowGc::AlphaBlendBitmaps(const TPoint& aDestPt, const CWsBitmap* aSrcBmp, const TRect& aSrcRect,const CWsBitmap* aAlphaBmp, const TPoint& aAlphaPt)
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.
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.
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()
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());
1964 This method has been deprecated. Calling it has no effect.
1965 @param aDrawOpaque Ignored.
1968 EXPORT_C void CWindowGc::SetOpaque(TBool aDrawOpaque)
1970 iPimpl->WriteAnyPendingStateChanges();
1971 WriteInt(aDrawOpaque, EWsGcOpSetOpaque);
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
1987 @param aInput is a TAny pointer used to input data.
1989 EXPORT_C TInt CWindowGc::APIExtension(TUid aUid, TAny*& aOutput, TAny* aInput)
1991 if (aUid == KGetUnderlineMetrics)
1993 return APIExGetUnderlineMetrics(aOutput);
1995 else if (aUid == KSetShadowColor)
1997 return APIExSetShadowColor(aInput);
1999 else if (aUid == KGetShadowColor)
2001 return APIExGetShadowColor(aOutput);
2003 else if (aUid == KDrawTextInContextUid)
2005 TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
2006 return APIExDrawText(contextParam->iText, &contextParam->iParam, contextParam->iPosition);
2008 else if (aUid == KDrawBoxTextInContextUid)
2010 TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
2011 return APIExDrawText(contextParam->iText,&contextParam->iParam,contextParam->iBox,contextParam->iBaselineOffset,contextParam->iAlign,contextParam->iMargin);
2013 else if (aUid == KDrawTextInContextVerticalUid)
2015 TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
2016 return APIExDrawTextVertical(contextParam->iText, &contextParam->iParam, contextParam->iPosition,contextParam->iUp);
2018 else if (aUid == KDrawBoxTextInContextVerticalUid)
2020 TDrawTextInContextInternal* contextParam = (TDrawTextInContextInternal*)aInput;
2021 return APIExDrawTextVertical(contextParam->iText,&contextParam->iParam,contextParam->iBox,contextParam->iBaselineOffset,contextParam->iUp,contextParam->iAlign,contextParam->iMargin);
2023 else if (aUid == KApiExtensionInterfaceUid)
2025 return APIExInterface(aOutput, *static_cast<TUid*>(aInput));
2027 /* Future cases may be placed here later.*/
2029 return CBitmapContext::APIExtension(aUid, aOutput, aInput);
2032 //The methods listed above in APIExtension follow here with the prefix APIEx.
2033 TInt CWindowGc::APIExGetUnderlineMetrics(TAny*& aOutput)
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;
2042 TInt CWindowGc::APIExSetShadowColor(TAny* aShadowColor)
2044 const TRgb shadowColor = *(reinterpret_cast<TRgb*> (aShadowColor));
2045 WriteInt(shadowColor.Internal(), EWsGcOpSetShadowColor);
2046 iPimpl->iShadowColor = shadowColor;
2050 TInt CWindowGc::APIExGetShadowColor(TAny*& aOutput)
2052 TRgb* ptr = (TRgb*)aOutput;
2053 ptr->SetInternal(iPimpl->iShadowColor.Internal());
2057 //Default implementation of reserved virtual
2058 EXPORT_C void CWindowGc::Reserved_CGraphicsContext_2()
2060 CBitmapContext::Reserved_CGraphicsContext_2();
2063 //Default implementation of reserved virtual
2064 EXPORT_C void CWindowGc::Reserved_CBitmapContext_1()
2066 CBitmapContext::Reserved_CBitmapContext_1();
2069 //Default implementation of reserved virtual
2070 EXPORT_C void CWindowGc::Reserved_CBitmapContext_2()
2072 CBitmapContext::Reserved_CBitmapContext_2();
2075 //Default implementation of reserved virtual
2076 EXPORT_C void CWindowGc::Reserved_CBitmapContext_3()
2078 CBitmapContext::Reserved_CBitmapContext_3();
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.
2086 @param aId the identifier for the artwork
2087 @param aDestRect the destination rect within the active window for this artwork
2093 iPimpl->WriteAnyPendingStateChanges();
2094 TWsGcCmdDrawWsGraphic drawWsGraphic(aId,aDestRect);
2095 Write(&drawWsGraphic,sizeof(drawWsGraphic),EWsGcOpDrawWsGraphic);
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.
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
2111 iPimpl->WriteAnyPendingStateChanges();
2112 TWsGcCmdDrawWsGraphic drawWsGraphic(aId,aDestRect);
2113 drawWsGraphic.iDataLen = aData.Size();
2114 WriteTextCommand(&drawWsGraphic, sizeof(drawWsGraphic), aData, EWsGcOpDrawWsGraphic, EWsGcOpDrawWsGraphicPtr);
2118 Gets an extension interface specified by the supplied UID, or NULL if it isn't supported.
2120 @param aInterfaceId The UID of the requested interface
2121 @return A pointer to the interface, or NULL if the interface isn't supported
2125 EXPORT_C TAny* CWindowGc::Interface(TUid aInterfaceId)
2127 TAny* interface = NULL;
2128 if(KErrNone == APIExtension(KApiExtensionInterfaceUid, interface, &aInterfaceId))
2134 Gets an extension interface specified by the supplied UID, or NULL if it isn't supported.
2136 @param aInterfaceId The UID of the requested interface
2137 @return A pointer to the interface, or NULL if the interface isn't supported
2141 EXPORT_C const TAny* CWindowGc::Interface(TUid aInterfaceId) const
2143 return const_cast<CWindowGc*>(this)->Interface(aInterfaceId);
2146 TInt CWindowGc::APIExInterface(TAny*& aInterface, TUid aInterfaceId)
2148 if(aInterfaceId == KMWsDrawResourceInterfaceUid)
2150 aInterface = static_cast<MWsDrawResource*>(iPimpl);
2153 return KErrNotSupported;
2156 void CWindowGc::DrawResource(const TPoint& aPos, const RWsDrawableSource& aSource, TGraphicsRotation aRotation)
2158 iPimpl->WriteAnyPendingStateChanges();
2159 TWsGcCmdDrawResourceToPos drawWsResource(aSource.WsHandle(), aPos, aRotation);
2160 Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceToPos);
2163 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, TGraphicsRotation aRotation)
2165 iPimpl->WriteAnyPendingStateChanges();
2166 TWsGcCmdDrawResourceToRect drawWsResource(aSource.WsHandle(), aDestRect, aRotation);
2167 Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceToRect);
2170 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TRect& aSrcRect, TGraphicsRotation aRotation)
2172 iPimpl->WriteAnyPendingStateChanges();
2173 TWsGcCmdDrawResourceFromRectToRect drawWsResource(aSource.WsHandle(), aDestRect, aSrcRect, aRotation);
2174 Write(&drawWsResource, sizeof(drawWsResource), EWsGcOpDrawResourceFromRectToRect);
2177 void CWindowGc::DrawResource(const TRect& aDestRect, const RWsDrawableSource& aSource, const TDesC8& aParam)
2179 iPimpl->WriteAnyPendingStateChanges();
2180 TWsGcCmdDrawResourceWithData drawWsResource(aSource.WsHandle(), aDestRect, &aParam);
2181 Write(&drawWsResource, sizeof(drawWsResource),EWsGcOpDrawResourceWithData);
2184 //Default implementation of reserved virtual
2185 EXPORT_C void CWindowGc::Reserved_CWindowGc_3()
2189 //Default implementation of reserved virtual
2190 EXPORT_C void CWindowGc::Reserved_CWindowGc_4()
2194 //Default implementation of reserved virtual
2195 EXPORT_C void CWindowGc::Reserved_CWindowGc_5()
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.
2204 EXPORT_C RWsDrawableSource::RWsDrawableSource()
2205 : iDrawableId(KSgNullDrawableId), iScreenNumber(KSgScreenIdMain)
2211 @param aWs Session to the window server
2213 @pre Connection to the window server is established
2215 EXPORT_C RWsDrawableSource::RWsDrawableSource(RWsSession &aWs)
2216 : MWsClientClass(aWs.iBuffer), iDrawableId(KSgNullDrawableId), iScreenNumber(KSgScreenIdMain)
2221 Create window server object for resource drawing operation via window server.
2223 This object will be identified by a unique handle and will be associated with drawable resource which is passed as a parameter.
2225 This object will be created for drawing onto the default screen only.
2228 @param aDrawable Drawable resource.
2230 @post Drawable source is created and can be used by window server. The reference counter of the underlying
2231 image resource is incremented.
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.
2237 EXPORT_C TInt RWsDrawableSource::Create(const RSgDrawable& aDrawable)
2239 return Create(aDrawable, KSgScreenIdMain);
2243 Create window server object for resource drawing operation via window server.
2245 This object will be identified by unique handle and will be associated with drawable resource which is passed as a parameter.
2247 This object will be created for drawing onto the specified screen only.
2250 @param aDrawable Drawable resource.
2251 @param aScreenNumber The screen onto which this drawable resource can be drawn.
2253 @post Drawable source is created and can be used by window server. The reference counter of the underlying
2254 image resource is incremented.
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.
2261 EXPORT_C TInt RWsDrawableSource::Create(const RSgDrawable& aDrawable, TInt aScreenNumber)
2265 return KErrAlreadyExists;
2267 CGraphicsResourceWrapperFactory* grwFactory = new CGraphicsResourceWrapperFactory();
2269 return KErrNoMemory;
2270 // coverity[uninit_use_in_call]
2271 CGraphicsResourceWrapper* graphicsResource = grwFactory->NewGraphicsResourceWrapper();
2272 if(!graphicsResource)
2275 return KErrNotSupported;
2277 if (graphicsResource->IsNull(aDrawable) || aScreenNumber < 0)
2279 delete graphicsResource;
2281 return KErrArgument;
2283 TWsClCmdCreateDrawableSource create(graphicsResource->Id(aDrawable), aScreenNumber);
2285 if ((ret = iBuffer->WriteReplyWs(&create, sizeof(TWsClCmdCreateDrawableSource), EWsClOpCreateDrawableSource)) < 0)
2287 delete graphicsResource;
2292 iDrawableId = graphicsResource->Id(aDrawable);
2293 iScreenNumber = aScreenNumber;
2294 delete graphicsResource;
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.
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.
2308 EXPORT_C void RWsDrawableSource::Close()
2312 Write(EWsDrawableSourceOpFree);
2314 iDrawableId = KSgNullDrawableId;
2315 iScreenNumber = KSgScreenIdMain;
2320 Get the unique ID of the associated drawable resource.
2322 EXPORT_C const TSgDrawableId& RWsDrawableSource::DrawableId() const
2328 Get the screen number of the drawable source.
2330 EXPORT_C TInt RWsDrawableSource::ScreenNumber() const
2332 return iScreenNumber;