1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/graphics.cpp
3 // Purpose: wxGCDC class
4 // Author: Stefan Csomor
8 // Copyright: (c) 2006 Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
20 #if wxUSE_GRAPHICS_CONTEXT
23 #include "wx/msw/wrapcdlg.h"
25 #include "wx/window.h"
27 #include "wx/dialog.h"
29 #include "wx/bitmap.h"
32 #include "wx/module.h"
33 // include all dc types that are used as a param
35 #include "wx/dcclient.h"
36 #include "wx/dcmemory.h"
37 #include "wx/dcprint.h"
42 #include "wx/private/graphics.h"
43 #include "wx/msw/wrapgdip.h"
44 #include "wx/msw/dc.h"
45 #if wxUSE_ENH_METAFILE
46 #include "wx/msw/enhmeta.h"
48 #include "wx/dcgraph.h"
50 #include "wx/msw/private.h" // needs to be before #include <commdlg.h>
52 #if wxUSE_COMMON_DIALOGS && !defined(__WXMICROWIN__)
59 //-----------------------------------------------------------------------------
61 //-----------------------------------------------------------------------------
63 const double RAD2DEG
= 180.0 / M_PI
;
65 //-----------------------------------------------------------------------------
67 //-----------------------------------------------------------------------------
69 inline double dmin(double a
, double b
) { return a
< b
? a
: b
; }
70 inline double dmax(double a
, double b
) { return a
> b
? a
: b
; }
72 inline double DegToRad(double deg
) { return (deg
* M_PI
) / 180.0; }
73 inline double RadToDeg(double deg
) { return (deg
* 180.0) / M_PI
; }
75 // translate a wxColour to a Color
76 inline Color
wxColourToColor(const wxColour
& col
)
78 return Color(col
.Alpha(), col
.Red(), col
.Green(), col
.Blue());
81 // Do not use this pointer directly, it's only used by
82 // GetDrawTextStringFormat() and the cleanup code in wxGDIPlusRendererModule.
83 StringFormat
* gs_drawTextStringFormat
= NULL
;
85 // Get the string format used for the text drawing and measuring functions:
86 // notice that it must be the same one for all of them, otherwise the drawn
87 // text might be of different size than what measuring it returned.
88 inline StringFormat
* GetDrawTextStringFormat()
90 if ( !gs_drawTextStringFormat
)
92 // We create this string format with exactly the same flags as
93 // StringFormat::GenericTypographic() is documented to use in MSDN
94 // except for the last one which doesn't make any difference for
95 // DrawText() but that we do want to use when measuring text.
97 // The reason for not just using GenericTypographic itself is that it
98 // does something else (what exactly is unfortunately not documented),
99 // which results in string being displayed quite differently from the
100 // default rendering, see #14537.
101 gs_drawTextStringFormat
104 StringFormatFlagsLineLimit
|
105 StringFormatFlagsNoClip
|
106 StringFormatFlagsNoFitBlackBox
|
107 StringFormatFlagsMeasureTrailingSpaces
111 return gs_drawTextStringFormat
;
114 } // anonymous namespace
116 //-----------------------------------------------------------------------------
117 // device context implementation
119 // more and more of the dc functionality should be implemented by calling
120 // the appropricate wxGDIPlusContext, but we will have to do that step by step
121 // also coordinate conversions should be moved to native matrix ops
122 //-----------------------------------------------------------------------------
124 // we always stock two context states, one at entry, to be able to preserve the
125 // state we were called with, the other one after changing to HI Graphics orientation
126 // (this one is used for getting back clippings etc)
128 //-----------------------------------------------------------------------------
129 // wxGraphicsPath implementation
130 //-----------------------------------------------------------------------------
132 class wxGDIPlusContext
;
134 class wxGDIPlusPathData
: public wxGraphicsPathData
137 wxGDIPlusPathData(wxGraphicsRenderer
* renderer
, GraphicsPath
* path
= NULL
);
138 ~wxGDIPlusPathData();
140 virtual wxGraphicsObjectRefData
*Clone() const;
143 // These are the path primitives from which everything else can be constructed
146 // begins a new subpath at (x,y)
147 virtual void MoveToPoint( wxDouble x
, wxDouble y
);
149 // adds a straight line from the current point to (x,y)
150 virtual void AddLineToPoint( wxDouble x
, wxDouble y
);
152 // adds a cubic Bezier curve from the current point, using two control points and an end point
153 virtual void AddCurveToPoint( wxDouble cx1
, wxDouble cy1
, wxDouble cx2
, wxDouble cy2
, wxDouble x
, wxDouble y
);
156 // adds an arc of a circle centering at (x,y) with radius (r) from startAngle to endAngle
157 virtual void AddArc( wxDouble x
, wxDouble y
, wxDouble r
, wxDouble startAngle
, wxDouble endAngle
, bool clockwise
) ;
159 // gets the last point of the current path, (0,0) if not yet set
160 virtual void GetCurrentPoint( wxDouble
* x
, wxDouble
* y
) const;
163 virtual void AddPath( const wxGraphicsPathData
* path
);
165 // closes the current sub-path
166 virtual void CloseSubpath();
169 // These are convenience functions which - if not available natively will be assembled
170 // using the primitives from above
173 // appends a rectangle as a new closed subpath
174 virtual void AddRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
) ;
177 // appends an ellipsis as a new closed subpath fitting the passed rectangle
178 virtual void AddEllipsis( wxDouble x, wxDouble y, wxDouble w , wxDouble h ) ;
180 // draws a an arc to two tangents connecting (current) to (x1,y1) and (x1,y1) to (x2,y2), also a straight line from (current) to (x1,y1)
181 virtual void AddArcToPoint( wxDouble x1, wxDouble y1 , wxDouble x2, wxDouble y2, wxDouble r ) ;
184 // returns the native path
185 virtual void * GetNativePath() const { return m_path
; }
187 // give the native path returned by GetNativePath() back (there might be some deallocations necessary)
188 virtual void UnGetNativePath(void * WXUNUSED(path
)) const {}
190 // transforms each point of this path by the matrix
191 virtual void Transform( const wxGraphicsMatrixData
* matrix
) ;
193 // gets the bounding box enclosing all points (possibly including control points)
194 virtual void GetBox(wxDouble
*x
, wxDouble
*y
, wxDouble
*w
, wxDouble
*h
) const;
196 virtual bool Contains( wxDouble x
, wxDouble y
, wxPolygonFillMode fillStyle
= wxODDEVEN_RULE
) const;
199 GraphicsPath
* m_path
;
202 class wxGDIPlusMatrixData
: public wxGraphicsMatrixData
205 wxGDIPlusMatrixData(wxGraphicsRenderer
* renderer
, Matrix
* matrix
= NULL
) ;
206 virtual ~wxGDIPlusMatrixData() ;
208 virtual wxGraphicsObjectRefData
* Clone() const ;
210 // concatenates the matrix
211 virtual void Concat( const wxGraphicsMatrixData
*t
);
213 // sets the matrix to the respective values
214 virtual void Set(wxDouble a
=1.0, wxDouble b
=0.0, wxDouble c
=0.0, wxDouble d
=1.0,
215 wxDouble tx
=0.0, wxDouble ty
=0.0);
217 // gets the component valuess of the matrix
218 virtual void Get(wxDouble
* a
=NULL
, wxDouble
* b
=NULL
, wxDouble
* c
=NULL
,
219 wxDouble
* d
=NULL
, wxDouble
* tx
=NULL
, wxDouble
* ty
=NULL
) const;
221 // makes this the inverse matrix
222 virtual void Invert();
224 // returns true if the elements of the transformation matrix are equal ?
225 virtual bool IsEqual( const wxGraphicsMatrixData
* t
) const ;
227 // return true if this is the identity matrix
228 virtual bool IsIdentity() const;
234 // add the translation to this matrix
235 virtual void Translate( wxDouble dx
, wxDouble dy
);
237 // add the scale to this matrix
238 virtual void Scale( wxDouble xScale
, wxDouble yScale
);
240 // add the rotation to this matrix (radians)
241 virtual void Rotate( wxDouble angle
);
244 // apply the transforms
247 // applies that matrix to the point
248 virtual void TransformPoint( wxDouble
*x
, wxDouble
*y
) const;
250 // applies the matrix except for translations
251 virtual void TransformDistance( wxDouble
*dx
, wxDouble
*dy
) const;
253 // returns the native representation
254 virtual void * GetNativeMatrix() const;
259 class wxGDIPlusPenData
: public wxGraphicsObjectRefData
262 wxGDIPlusPenData( wxGraphicsRenderer
* renderer
, const wxPen
&pen
);
267 virtual wxDouble
GetWidth() { return m_width
; }
268 virtual Pen
* GetGDIPlusPen() { return m_pen
; }
278 class wxGDIPlusBrushData
: public wxGraphicsObjectRefData
281 wxGDIPlusBrushData( wxGraphicsRenderer
* renderer
);
282 wxGDIPlusBrushData( wxGraphicsRenderer
* renderer
, const wxBrush
&brush
);
283 ~wxGDIPlusBrushData ();
285 void CreateLinearGradientBrush(wxDouble x1
, wxDouble y1
,
286 wxDouble x2
, wxDouble y2
,
287 const wxGraphicsGradientStops
& stops
);
288 void CreateRadialGradientBrush(wxDouble xo
, wxDouble yo
,
289 wxDouble xc
, wxDouble yc
,
291 const wxGraphicsGradientStops
& stops
);
293 virtual Brush
* GetGDIPlusBrush() { return m_brush
; }
299 // common part of Create{Linear,Radial}GradientBrush()
300 template <typename T
>
301 void SetGradientStops(T
*brush
, const wxGraphicsGradientStops
& stops
);
305 GraphicsPath
* m_brushPath
;
308 class WXDLLIMPEXP_CORE wxGDIPlusBitmapData
: public wxGraphicsBitmapData
311 wxGDIPlusBitmapData( wxGraphicsRenderer
* renderer
, Bitmap
* bitmap
);
312 wxGDIPlusBitmapData( wxGraphicsRenderer
* renderer
, const wxBitmap
&bmp
);
313 ~wxGDIPlusBitmapData ();
315 virtual Bitmap
* GetGDIPlusBitmap() { return m_bitmap
; }
316 virtual void* GetNativeBitmap() const { return m_bitmap
; }
319 wxImage
ConvertToImage() const;
320 #endif // wxUSE_IMAGE
327 class wxGDIPlusFontData
: public wxGraphicsObjectRefData
330 wxGDIPlusFontData( wxGraphicsRenderer
* renderer
,
332 const wxColour
& col
);
333 wxGDIPlusFontData(wxGraphicsRenderer
* renderer
,
334 const wxString
& name
,
337 const wxColour
& col
);
338 ~wxGDIPlusFontData();
340 virtual Brush
* GetGDIPlusBrush() { return m_textBrush
; }
341 virtual Font
* GetGDIPlusFont() { return m_font
; }
344 // Common part of all ctors, flags here is a combination of values of
345 // FontStyle GDI+ enum.
346 void Init(const wxString
& name
,
356 class wxGDIPlusContext
: public wxGraphicsContext
359 wxGDIPlusContext( wxGraphicsRenderer
* renderer
, const wxDC
& dc
);
360 wxGDIPlusContext( wxGraphicsRenderer
* renderer
, HDC hdc
, wxDouble width
, wxDouble height
);
361 wxGDIPlusContext( wxGraphicsRenderer
* renderer
, HWND hwnd
);
362 wxGDIPlusContext( wxGraphicsRenderer
* renderer
, Graphics
* gr
);
363 wxGDIPlusContext(wxGraphicsRenderer
* renderer
);
365 virtual ~wxGDIPlusContext();
367 virtual void Clip( const wxRegion
®ion
);
368 // clips drawings to the rect
369 virtual void Clip( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
371 // resets the clipping to original extent
372 virtual void ResetClip();
374 virtual void * GetNativeContext();
376 virtual void StrokePath( const wxGraphicsPath
& p
);
377 virtual void FillPath( const wxGraphicsPath
& p
, wxPolygonFillMode fillStyle
= wxODDEVEN_RULE
);
379 virtual void DrawRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
381 // stroke lines connecting each of the points
382 virtual void StrokeLines( size_t n
, const wxPoint2DDouble
*points
);
384 // We don't have any specific implementation for this one in wxMSW but
385 // override it just to avoid warnings about hiding the base class virtual.
386 virtual void StrokeLines( size_t n
, const wxPoint2DDouble
*beginPoints
, const wxPoint2DDouble
*endPoints
)
388 wxGraphicsContext::StrokeLines(n
, beginPoints
, endPoints
);
392 virtual void DrawLines( size_t n
, const wxPoint2DDouble
*points
, wxPolygonFillMode fillStyle
= wxODDEVEN_RULE
);
394 virtual bool SetAntialiasMode(wxAntialiasMode antialias
);
396 virtual bool SetInterpolationQuality(wxInterpolationQuality interpolation
);
398 virtual bool SetCompositionMode(wxCompositionMode op
);
400 virtual void BeginLayer(wxDouble opacity
);
402 virtual void EndLayer();
404 virtual void Translate( wxDouble dx
, wxDouble dy
);
405 virtual void Scale( wxDouble xScale
, wxDouble yScale
);
406 virtual void Rotate( wxDouble angle
);
408 // concatenates this transform with the current transform of this context
409 virtual void ConcatTransform( const wxGraphicsMatrix
& matrix
);
411 // sets the transform of this context
412 virtual void SetTransform( const wxGraphicsMatrix
& matrix
);
414 // gets the matrix of this context
415 virtual wxGraphicsMatrix
GetTransform() const;
417 virtual void DrawBitmap( const wxGraphicsBitmap
&bmp
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
418 virtual void DrawBitmap( const wxBitmap
&bmp
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
419 virtual void DrawIcon( const wxIcon
&icon
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
420 virtual void PushState();
421 virtual void PopState();
423 virtual void GetTextExtent( const wxString
&str
, wxDouble
*width
, wxDouble
*height
,
424 wxDouble
*descent
, wxDouble
*externalLeading
) const;
425 virtual void GetPartialTextExtents(const wxString
& text
, wxArrayDouble
& widths
) const;
426 virtual bool ShouldOffset() const;
427 virtual void GetSize( wxDouble
* width
, wxDouble
*height
);
429 Graphics
* GetGraphics() const { return m_context
; }
433 wxDouble m_fontScaleRatio
;
435 // Used from ctors (including those in the derived classes) and takes
436 // ownership of the graphics pointer that must be non-NULL.
437 void Init(Graphics
* graphics
, int width
, int height
);
440 virtual void DoDrawText(const wxString
& str
, wxDouble x
, wxDouble y
);
443 wxStack
<GraphicsState
> m_stateStack
;
444 GraphicsState m_state1
;
445 GraphicsState m_state2
;
447 wxDECLARE_NO_COPY_CLASS(wxGDIPlusContext
);
452 class wxGDIPlusImageContext
: public wxGDIPlusContext
455 wxGDIPlusImageContext(wxGraphicsRenderer
* renderer
, wxImage
& image
) :
456 wxGDIPlusContext(renderer
),
458 m_bitmap(renderer
, image
)
462 new Graphics(m_bitmap
.GetGDIPlusBitmap()),
468 virtual ~wxGDIPlusImageContext()
470 m_image
= m_bitmap
.ConvertToImage();
475 wxGDIPlusBitmapData m_bitmap
;
477 wxDECLARE_NO_COPY_CLASS(wxGDIPlusImageContext
);
480 #endif // wxUSE_IMAGE
482 class wxGDIPlusMeasuringContext
: public wxGDIPlusContext
485 wxGDIPlusMeasuringContext( wxGraphicsRenderer
* renderer
) : wxGDIPlusContext( renderer
, m_hdc
= GetDC(NULL
), 1000, 1000 )
489 virtual ~wxGDIPlusMeasuringContext()
491 ReleaseDC( NULL
, m_hdc
);
498 class wxGDIPlusPrintingContext
: public wxGDIPlusContext
501 wxGDIPlusPrintingContext( wxGraphicsRenderer
* renderer
, const wxDC
& dc
);
502 virtual ~wxGDIPlusPrintingContext() { }
506 //-----------------------------------------------------------------------------
507 // wxGDIPlusRenderer declaration
508 //-----------------------------------------------------------------------------
510 class wxGDIPlusRenderer
: public wxGraphicsRenderer
519 virtual ~wxGDIPlusRenderer()
529 virtual wxGraphicsContext
* CreateContext( const wxWindowDC
& dc
);
531 virtual wxGraphicsContext
* CreateContext( const wxMemoryDC
& dc
);
533 #if wxUSE_PRINTING_ARCHITECTURE
534 virtual wxGraphicsContext
* CreateContext( const wxPrinterDC
& dc
);
537 #if wxUSE_ENH_METAFILE
538 virtual wxGraphicsContext
* CreateContext( const wxEnhMetaFileDC
& dc
);
541 virtual wxGraphicsContext
* CreateContextFromNativeContext( void * context
);
543 virtual wxGraphicsContext
* CreateContextFromNativeWindow( void * window
);
545 virtual wxGraphicsContext
* CreateContext( wxWindow
* window
);
548 virtual wxGraphicsContext
* CreateContextFromImage(wxImage
& image
);
549 #endif // wxUSE_IMAGE
551 virtual wxGraphicsContext
* CreateMeasuringContext();
555 virtual wxGraphicsPath
CreatePath();
559 virtual wxGraphicsMatrix
CreateMatrix( wxDouble a
=1.0, wxDouble b
=0.0, wxDouble c
=0.0, wxDouble d
=1.0,
560 wxDouble tx
=0.0, wxDouble ty
=0.0);
563 virtual wxGraphicsPen
CreatePen(const wxPen
& pen
) ;
565 virtual wxGraphicsBrush
CreateBrush(const wxBrush
& brush
) ;
567 virtual wxGraphicsBrush
568 CreateLinearGradientBrush(wxDouble x1
, wxDouble y1
,
569 wxDouble x2
, wxDouble y2
,
570 const wxGraphicsGradientStops
& stops
);
572 virtual wxGraphicsBrush
573 CreateRadialGradientBrush(wxDouble xo
, wxDouble yo
,
574 wxDouble xc
, wxDouble yc
,
576 const wxGraphicsGradientStops
& stops
);
578 // create a native bitmap representation
579 virtual wxGraphicsBitmap
CreateBitmap( const wxBitmap
&bitmap
);
581 virtual wxGraphicsBitmap
CreateBitmapFromImage(const wxImage
& image
);
582 virtual wxImage
CreateImageFromBitmap(const wxGraphicsBitmap
& bmp
);
583 #endif // wxUSE_IMAGE
585 virtual wxGraphicsFont
CreateFont( const wxFont
& font
,
586 const wxColour
& col
);
588 virtual wxGraphicsFont
CreateFont(double size
,
589 const wxString
& facename
,
590 int flags
= wxFONTFLAG_DEFAULT
,
591 const wxColour
& col
= *wxBLACK
);
593 // create a graphics bitmap from a native bitmap
594 virtual wxGraphicsBitmap
CreateBitmapFromNativeBitmap( void* bitmap
);
596 // create a subimage from a native image representation
597 virtual wxGraphicsBitmap
CreateSubBitmap( const wxGraphicsBitmap
&bitmap
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
);
600 bool EnsureIsLoaded();
603 friend class wxGDIPlusRendererModule
;
607 ULONG_PTR m_gditoken
;
609 DECLARE_DYNAMIC_CLASS_NO_COPY(wxGDIPlusRenderer
)
612 //-----------------------------------------------------------------------------
613 // wxGDIPlusPen implementation
614 //-----------------------------------------------------------------------------
616 wxGDIPlusPenData::~wxGDIPlusPenData()
623 void wxGDIPlusPenData::Init()
630 wxGDIPlusPenData::wxGDIPlusPenData( wxGraphicsRenderer
* renderer
, const wxPen
&pen
)
631 : wxGraphicsObjectRefData(renderer
)
634 m_width
= pen
.GetWidth();
638 m_pen
= new Pen(wxColourToColor(pen
.GetColour()), m_width
);
641 switch ( pen
.GetCap() )
647 case wxCAP_PROJECTING
:
652 cap
= LineCapFlat
; // TODO verify
659 m_pen
->SetLineCap(cap
,cap
, DashCapFlat
);
662 switch ( pen
.GetJoin() )
665 join
= LineJoinBevel
;
669 join
= LineJoinMiter
;
673 join
= LineJoinRound
;
677 join
= LineJoinMiter
;
681 m_pen
->SetLineJoin(join
);
683 m_pen
->SetDashStyle(DashStyleSolid
);
685 DashStyle dashStyle
= DashStyleSolid
;
686 switch ( pen
.GetStyle() )
688 case wxPENSTYLE_SOLID
:
691 case wxPENSTYLE_DOT
:
692 dashStyle
= DashStyleDot
;
695 case wxPENSTYLE_LONG_DASH
:
696 dashStyle
= DashStyleDash
; // TODO verify
699 case wxPENSTYLE_SHORT_DASH
:
700 dashStyle
= DashStyleDash
;
703 case wxPENSTYLE_DOT_DASH
:
704 dashStyle
= DashStyleDashDot
;
706 case wxPENSTYLE_USER_DASH
:
708 dashStyle
= DashStyleCustom
;
710 int count
= pen
.GetDashes( &dashes
);
711 if ((dashes
!= NULL
) && (count
> 0))
713 REAL
*userLengths
= new REAL
[count
];
714 for ( int i
= 0; i
< count
; ++i
)
716 userLengths
[i
] = dashes
[i
];
718 m_pen
->SetDashPattern( userLengths
, count
);
719 delete[] userLengths
;
723 case wxPENSTYLE_STIPPLE
:
725 wxBitmap
* bmp
= pen
.GetStipple();
726 if ( bmp
&& bmp
->IsOk() )
728 m_penImage
= Bitmap::FromHBITMAP((HBITMAP
)bmp
->GetHBITMAP(),
730 (HPALETTE
)bmp
->GetPalette()->GetHPALETTE()
735 m_penBrush
= new TextureBrush(m_penImage
);
736 m_pen
->SetBrush( m_penBrush
);
742 if ( pen
.GetStyle() >= wxPENSTYLE_FIRST_HATCH
&&
743 pen
.GetStyle() <= wxPENSTYLE_LAST_HATCH
)
746 switch( pen
.GetStyle() )
748 case wxPENSTYLE_BDIAGONAL_HATCH
:
749 style
= HatchStyleBackwardDiagonal
;
751 case wxPENSTYLE_CROSSDIAG_HATCH
:
752 style
= HatchStyleDiagonalCross
;
754 case wxPENSTYLE_FDIAGONAL_HATCH
:
755 style
= HatchStyleForwardDiagonal
;
757 case wxPENSTYLE_CROSS_HATCH
:
758 style
= HatchStyleCross
;
760 case wxPENSTYLE_HORIZONTAL_HATCH
:
761 style
= HatchStyleHorizontal
;
763 case wxPENSTYLE_VERTICAL_HATCH
:
764 style
= HatchStyleVertical
;
767 style
= HatchStyleHorizontal
;
769 m_penBrush
= new HatchBrush
772 wxColourToColor(pen
.GetColour()),
775 m_pen
->SetBrush( m_penBrush
);
779 if ( dashStyle
!= DashStyleSolid
)
780 m_pen
->SetDashStyle(dashStyle
);
783 //-----------------------------------------------------------------------------
784 // wxGDIPlusBrush implementation
785 //-----------------------------------------------------------------------------
787 wxGDIPlusBrushData::wxGDIPlusBrushData( wxGraphicsRenderer
* renderer
)
788 : wxGraphicsObjectRefData(renderer
)
793 wxGDIPlusBrushData::wxGDIPlusBrushData( wxGraphicsRenderer
* renderer
, const wxBrush
&brush
)
794 : wxGraphicsObjectRefData(renderer
)
797 if ( brush
.GetStyle() == wxSOLID
)
799 m_brush
= new SolidBrush(wxColourToColor( brush
.GetColour()));
801 else if ( brush
.IsHatch() )
804 switch( brush
.GetStyle() )
806 case wxBRUSHSTYLE_BDIAGONAL_HATCH
:
807 style
= HatchStyleBackwardDiagonal
;
809 case wxBRUSHSTYLE_CROSSDIAG_HATCH
:
810 style
= HatchStyleDiagonalCross
;
812 case wxBRUSHSTYLE_FDIAGONAL_HATCH
:
813 style
= HatchStyleForwardDiagonal
;
815 case wxBRUSHSTYLE_CROSS_HATCH
:
816 style
= HatchStyleCross
;
818 case wxBRUSHSTYLE_HORIZONTAL_HATCH
:
819 style
= HatchStyleHorizontal
;
821 case wxBRUSHSTYLE_VERTICAL_HATCH
:
822 style
= HatchStyleVertical
;
825 style
= HatchStyleHorizontal
;
827 m_brush
= new HatchBrush
830 wxColourToColor(brush
.GetColour()),
836 wxBitmap
* bmp
= brush
.GetStipple();
837 if ( bmp
&& bmp
->IsOk() )
839 wxDELETE( m_brushImage
);
840 m_brushImage
= Bitmap::FromHBITMAP((HBITMAP
)bmp
->GetHBITMAP(),
842 (HPALETTE
)bmp
->GetPalette()->GetHPALETTE()
847 m_brush
= new TextureBrush(m_brushImage
);
852 wxGDIPlusBrushData::~wxGDIPlusBrushData()
859 void wxGDIPlusBrushData::Init()
866 template <typename T
>
868 wxGDIPlusBrushData::SetGradientStops(T
*brush
,
869 const wxGraphicsGradientStops
& stops
)
871 const unsigned numStops
= stops
.GetCount();
874 // initial and final colours are set during the brush creation, nothing
879 wxVector
<Color
> colors(numStops
);
880 wxVector
<REAL
> positions(numStops
);
882 for ( unsigned i
= 0; i
< numStops
; i
++ )
884 wxGraphicsGradientStop stop
= stops
.Item(i
);
886 colors
[i
] = wxColourToColor(stop
.GetColour());
887 positions
[i
] = stop
.GetPosition();
890 brush
->SetInterpolationColors(&colors
[0], &positions
[0], numStops
);
894 wxGDIPlusBrushData::CreateLinearGradientBrush(wxDouble x1
, wxDouble y1
,
895 wxDouble x2
, wxDouble y2
,
896 const wxGraphicsGradientStops
& stops
)
898 LinearGradientBrush
* const
899 brush
= new LinearGradientBrush(PointF(x1
, y1
) , PointF(x2
, y2
),
900 wxColourToColor(stops
.GetStartColour()),
901 wxColourToColor(stops
.GetEndColour()));
904 SetGradientStops(brush
, stops
);
908 wxGDIPlusBrushData::CreateRadialGradientBrush(wxDouble xo
, wxDouble yo
,
909 wxDouble xc
, wxDouble yc
,
911 const wxGraphicsGradientStops
& stops
)
913 m_brushPath
= new GraphicsPath();
914 m_brushPath
->AddEllipse( (REAL
)(xc
-radius
), (REAL
)(yc
-radius
),
915 (REAL
)(2*radius
), (REAL
)(2*radius
));
917 PathGradientBrush
* const brush
= new PathGradientBrush(m_brushPath
);
919 brush
->SetCenterPoint(PointF(xo
, yo
));
920 brush
->SetCenterColor(wxColourToColor(stops
.GetStartColour()));
922 const Color
col(wxColourToColor(stops
.GetEndColour()));
924 brush
->SetSurroundColors(&col
, &count
);
926 SetGradientStops(brush
, stops
);
929 //-----------------------------------------------------------------------------
930 // wxGDIPlusFont implementation
931 //-----------------------------------------------------------------------------
934 wxGDIPlusFontData::Init(const wxString
& name
,
940 m_font
= new Font(name
.wc_str(), size
, style
, fontUnit
);
942 m_textBrush
= new SolidBrush(wxColourToColor(col
));
945 wxGDIPlusFontData::wxGDIPlusFontData( wxGraphicsRenderer
* renderer
,
947 const wxColour
& col
)
948 : wxGraphicsObjectRefData( renderer
)
950 int style
= FontStyleRegular
;
951 if ( font
.GetStyle() == wxFONTSTYLE_ITALIC
)
952 style
|= FontStyleItalic
;
953 if ( font
.GetUnderlined() )
954 style
|= FontStyleUnderline
;
955 if ( font
.GetWeight() == wxFONTWEIGHT_BOLD
)
956 style
|= FontStyleBold
;
958 Init(font
.GetFaceName(), font
.GetPointSize(), style
, col
, UnitPoint
);
961 wxGDIPlusFontData::wxGDIPlusFontData(wxGraphicsRenderer
* renderer
,
962 const wxString
& name
,
965 const wxColour
& col
) :
966 wxGraphicsObjectRefData(renderer
)
968 Init(name
, sizeInPixels
, style
, col
, UnitPixel
);
971 wxGDIPlusFontData::~wxGDIPlusFontData()
977 // the built-in conversions functions create non-premultiplied bitmaps, while GDIPlus needs them in the
978 // premultiplied format, therefore in the failing cases we create a new bitmap using the non-premultiplied
979 // bytes as parameter, since there is no real copying of the data going in, only references are stored
980 // m_helper has to be kept alive as well
982 //-----------------------------------------------------------------------------
983 // wxGDIPlusBitmapData implementation
984 //-----------------------------------------------------------------------------
986 wxGDIPlusBitmapData::wxGDIPlusBitmapData( wxGraphicsRenderer
* renderer
, Bitmap
* bitmap
) :
987 wxGraphicsBitmapData( renderer
), m_bitmap( bitmap
)
992 wxGDIPlusBitmapData::wxGDIPlusBitmapData( wxGraphicsRenderer
* renderer
,
993 const wxBitmap
&bmp
) : wxGraphicsBitmapData( renderer
)
998 Bitmap
* image
= NULL
;
1001 Bitmap
interim((HBITMAP
)bmp
.GetHBITMAP(),
1003 (HPALETTE
)bmp
.GetPalette()->GetHPALETTE()
1009 size_t width
= interim
.GetWidth();
1010 size_t height
= interim
.GetHeight();
1011 Rect
bounds(0,0,width
,height
);
1013 image
= new Bitmap(width
,height
,PixelFormat32bppPARGB
) ;
1015 Bitmap
interimMask((HBITMAP
)bmp
.GetMask()->GetMaskBitmap(),NULL
);
1016 wxASSERT(interimMask
.GetPixelFormat() == PixelFormat1bppIndexed
);
1018 BitmapData dataMask
;
1019 interimMask
.LockBits(&bounds
,ImageLockModeRead
,
1020 interimMask
.GetPixelFormat(),&dataMask
);
1023 BitmapData imageData
;
1024 image
->LockBits(&bounds
,ImageLockModeWrite
, PixelFormat32bppPARGB
, &imageData
);
1026 BYTE maskPattern
= 0 ;
1030 for ( size_t y
= 0 ; y
< height
; ++y
)
1033 for( size_t x
= 0 ; x
< width
; ++x
)
1038 maskByte
= *((BYTE
*)dataMask
.Scan0
+ dataMask
.Stride
*y
+ maskIndex
);
1042 maskPattern
= maskPattern
>> 1;
1044 ARGB
*dest
= (ARGB
*)((BYTE
*)imageData
.Scan0
+ imageData
.Stride
*y
+ x
*4);
1045 if ( (maskByte
& maskPattern
) == 0 )
1050 interim
.GetPixel(x
,y
,&c
) ;
1051 *dest
= (c
.GetValue() | Color::AlphaMask
);
1056 image
->UnlockBits(&imageData
);
1058 interimMask
.UnlockBits(&dataMask
);
1059 interim
.UnlockBits(&dataMask
);
1063 image
= Bitmap::FromHBITMAP((HBITMAP
)bmp
.GetHBITMAP(),
1065 (HPALETTE
)bmp
.GetPalette()->GetHPALETTE()
1070 if ( bmp
.HasAlpha() && GetPixelFormatSize(image
->GetPixelFormat()) == 32 )
1072 size_t width
= image
->GetWidth();
1073 size_t height
= image
->GetHeight();
1074 Rect
bounds(0,0,width
,height
);
1075 static BitmapData data
;
1079 m_helper
->LockBits(&bounds
, ImageLockModeRead
,
1080 m_helper
->GetPixelFormat(),&data
);
1082 image
= new Bitmap(data
.Width
, data
.Height
, data
.Stride
,
1083 PixelFormat32bppPARGB
, (BYTE
*) data
.Scan0
);
1085 m_helper
->UnlockBits(&data
);
1094 wxImage
wxGDIPlusBitmapData::ConvertToImage() const
1096 // We could use Bitmap::LockBits() and convert to wxImage directly but
1097 // passing by wxBitmap is easier. It would be nice to measure performance
1098 // of the two methods but for this the second one would need to be written
1101 if ( m_bitmap
->GetHBITMAP(Color(0xffffffff), &hbmp
) != Gdiplus::Ok
)
1105 bmp
.SetWidth(m_bitmap
->GetWidth());
1106 bmp
.SetHeight(m_bitmap
->GetHeight());
1107 bmp
.SetHBITMAP(hbmp
);
1108 bmp
.SetDepth(IsAlphaPixelFormat(m_bitmap
->GetPixelFormat()) ? 32 : 24);
1109 return bmp
.ConvertToImage();
1112 #endif // wxUSE_IMAGE
1114 wxGDIPlusBitmapData::~wxGDIPlusBitmapData()
1120 //-----------------------------------------------------------------------------
1121 // wxGDIPlusPath implementation
1122 //-----------------------------------------------------------------------------
1124 wxGDIPlusPathData::wxGDIPlusPathData(wxGraphicsRenderer
* renderer
, GraphicsPath
* path
) : wxGraphicsPathData(renderer
)
1129 m_path
= new GraphicsPath();
1132 wxGDIPlusPathData::~wxGDIPlusPathData()
1137 wxGraphicsObjectRefData
* wxGDIPlusPathData::Clone() const
1139 return new wxGDIPlusPathData( GetRenderer() , m_path
->Clone());
1146 void wxGDIPlusPathData::MoveToPoint( wxDouble x
, wxDouble y
)
1148 m_path
->StartFigure();
1149 m_path
->AddLine((REAL
) x
,(REAL
) y
,(REAL
) x
,(REAL
) y
);
1152 void wxGDIPlusPathData::AddLineToPoint( wxDouble x
, wxDouble y
)
1154 m_path
->AddLine((REAL
) x
,(REAL
) y
,(REAL
) x
,(REAL
) y
);
1157 void wxGDIPlusPathData::CloseSubpath()
1159 m_path
->CloseFigure();
1162 void wxGDIPlusPathData::AddCurveToPoint( wxDouble cx1
, wxDouble cy1
, wxDouble cx2
, wxDouble cy2
, wxDouble x
, wxDouble y
)
1168 m_path
->GetLastPoint(&start
);
1169 m_path
->AddBezier(start
,c1
,c2
,end
);
1172 // gets the last point of the current path, (0,0) if not yet set
1173 void wxGDIPlusPathData::GetCurrentPoint( wxDouble
* x
, wxDouble
* y
) const
1176 m_path
->GetLastPoint(&start
);
1181 void wxGDIPlusPathData::AddArc( wxDouble x
, wxDouble y
, wxDouble r
, double startAngle
, double endAngle
, bool clockwise
)
1183 double sweepAngle
= endAngle
- startAngle
;
1184 if( fabs(sweepAngle
) >= 2*M_PI
)
1186 sweepAngle
= 2 * M_PI
;
1192 if( sweepAngle
< 0 )
1193 sweepAngle
+= 2 * M_PI
;
1197 if( sweepAngle
> 0 )
1198 sweepAngle
-= 2 * M_PI
;
1202 m_path
->AddArc((REAL
) (x
-r
),(REAL
) (y
-r
),(REAL
) (2*r
),(REAL
) (2*r
),RadToDeg(startAngle
),RadToDeg(sweepAngle
));
1205 void wxGDIPlusPathData::AddRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1207 m_path
->AddRectangle(RectF(x
,y
,w
,h
));
1210 void wxGDIPlusPathData::AddPath( const wxGraphicsPathData
* path
)
1212 m_path
->AddPath( (GraphicsPath
*) path
->GetNativePath(), FALSE
);
1216 // transforms each point of this path by the matrix
1217 void wxGDIPlusPathData::Transform( const wxGraphicsMatrixData
* matrix
)
1219 m_path
->Transform( (Matrix
*) matrix
->GetNativeMatrix() );
1222 // gets the bounding box enclosing all points (possibly including control points)
1223 void wxGDIPlusPathData::GetBox(wxDouble
*x
, wxDouble
*y
, wxDouble
*w
, wxDouble
*h
) const
1226 m_path
->GetBounds( &bounds
, NULL
, NULL
) ;
1233 bool wxGDIPlusPathData::Contains( wxDouble x
, wxDouble y
, wxPolygonFillMode fillStyle
) const
1235 m_path
->SetFillMode( fillStyle
== wxODDEVEN_RULE
? FillModeAlternate
: FillModeWinding
);
1236 return m_path
->IsVisible( (FLOAT
) x
,(FLOAT
) y
) == TRUE
;
1239 //-----------------------------------------------------------------------------
1240 // wxGDIPlusMatrixData implementation
1241 //-----------------------------------------------------------------------------
1243 wxGDIPlusMatrixData::wxGDIPlusMatrixData(wxGraphicsRenderer
* renderer
, Matrix
* matrix
)
1244 : wxGraphicsMatrixData(renderer
)
1249 m_matrix
= new Matrix();
1252 wxGDIPlusMatrixData::~wxGDIPlusMatrixData()
1257 wxGraphicsObjectRefData
*wxGDIPlusMatrixData::Clone() const
1259 return new wxGDIPlusMatrixData( GetRenderer(), m_matrix
->Clone());
1262 // concatenates the matrix
1263 void wxGDIPlusMatrixData::Concat( const wxGraphicsMatrixData
*t
)
1265 m_matrix
->Multiply( (Matrix
*) t
->GetNativeMatrix());
1268 // sets the matrix to the respective values
1269 void wxGDIPlusMatrixData::Set(wxDouble a
, wxDouble b
, wxDouble c
, wxDouble d
,
1270 wxDouble tx
, wxDouble ty
)
1272 m_matrix
->SetElements(a
,b
,c
,d
,tx
,ty
);
1275 // gets the component valuess of the matrix
1276 void wxGDIPlusMatrixData::Get(wxDouble
* a
, wxDouble
* b
, wxDouble
* c
,
1277 wxDouble
* d
, wxDouble
* tx
, wxDouble
* ty
) const
1280 m_matrix
->GetElements(elements
);
1281 if (a
) *a
= elements
[0];
1282 if (b
) *b
= elements
[1];
1283 if (c
) *c
= elements
[2];
1284 if (d
) *d
= elements
[3];
1285 if (tx
) *tx
= elements
[4];
1286 if (ty
) *ty
= elements
[5];
1289 // makes this the inverse matrix
1290 void wxGDIPlusMatrixData::Invert()
1295 // returns true if the elements of the transformation matrix are equal ?
1296 bool wxGDIPlusMatrixData::IsEqual( const wxGraphicsMatrixData
* t
) const
1298 return m_matrix
->Equals((Matrix
*) t
->GetNativeMatrix())== TRUE
;
1301 // return true if this is the identity matrix
1302 bool wxGDIPlusMatrixData::IsIdentity() const
1304 return m_matrix
->IsIdentity() == TRUE
;
1311 // add the translation to this matrix
1312 void wxGDIPlusMatrixData::Translate( wxDouble dx
, wxDouble dy
)
1314 m_matrix
->Translate(dx
,dy
);
1317 // add the scale to this matrix
1318 void wxGDIPlusMatrixData::Scale( wxDouble xScale
, wxDouble yScale
)
1320 m_matrix
->Scale(xScale
,yScale
);
1323 // add the rotation to this matrix (radians)
1324 void wxGDIPlusMatrixData::Rotate( wxDouble angle
)
1326 m_matrix
->Rotate( RadToDeg(angle
) );
1330 // apply the transforms
1333 // applies that matrix to the point
1334 void wxGDIPlusMatrixData::TransformPoint( wxDouble
*x
, wxDouble
*y
) const
1337 m_matrix
->TransformPoints(&pt
);
1342 // applies the matrix except for translations
1343 void wxGDIPlusMatrixData::TransformDistance( wxDouble
*dx
, wxDouble
*dy
) const
1346 m_matrix
->TransformVectors(&pt
);
1351 // returns the native representation
1352 void * wxGDIPlusMatrixData::GetNativeMatrix() const
1357 //-----------------------------------------------------------------------------
1358 // wxGDIPlusContext implementation
1359 //-----------------------------------------------------------------------------
1361 class wxGDIPlusOffsetHelper
1364 wxGDIPlusOffsetHelper( Graphics
* gr
, bool offset
)
1369 m_gr
->TranslateTransform( 0.5, 0.5 );
1371 ~wxGDIPlusOffsetHelper( )
1374 m_gr
->TranslateTransform( -0.5, -0.5 );
1381 wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer
* renderer
, HDC hdc
, wxDouble width
, wxDouble height
)
1382 : wxGraphicsContext(renderer
)
1384 Init(new Graphics(hdc
), width
, height
);
1387 wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer
* renderer
, const wxDC
& dc
)
1388 : wxGraphicsContext(renderer
)
1390 wxMSWDCImpl
*msw
= wxDynamicCast( dc
.GetImpl() , wxMSWDCImpl
);
1391 HDC hdc
= (HDC
) msw
->GetHDC();
1392 wxSize sz
= dc
.GetSize();
1394 Init(new Graphics(hdc
), sz
.x
, sz
.y
);
1397 wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer
* renderer
, HWND hwnd
)
1398 : wxGraphicsContext(renderer
)
1400 RECT rect
= wxGetWindowRect(hwnd
);
1401 Init(new Graphics(hwnd
), rect
.right
- rect
.left
, rect
.bottom
- rect
.top
);
1402 m_enableOffset
= true;
1405 wxGDIPlusContext::wxGDIPlusContext( wxGraphicsRenderer
* renderer
, Graphics
* gr
)
1406 : wxGraphicsContext(renderer
)
1411 wxGDIPlusContext::wxGDIPlusContext(wxGraphicsRenderer
* renderer
)
1412 : wxGraphicsContext(renderer
)
1414 // Derived class must call Init() later but just set m_context to NULL for
1415 // safety to avoid crashing in our dtor if Init() ends up not being called.
1419 void wxGDIPlusContext::Init(Graphics
* graphics
, int width
, int height
)
1421 m_context
= graphics
;
1426 m_fontScaleRatio
= 1.0;
1428 m_context
->SetTextRenderingHint(TextRenderingHintSystemDefault
);
1429 m_context
->SetPixelOffsetMode(PixelOffsetModeHalf
);
1430 m_context
->SetSmoothingMode(SmoothingModeHighQuality
);
1431 m_context
->SetInterpolationMode(InterpolationModeHighQuality
);
1432 m_state1
= m_context
->Save();
1433 m_state2
= m_context
->Save();
1436 wxGDIPlusContext::~wxGDIPlusContext()
1440 m_context
->Restore( m_state2
);
1441 m_context
->Restore( m_state1
);
1447 void wxGDIPlusContext::Clip( const wxRegion
®ion
)
1449 Region
rgn((HRGN
)region
.GetHRGN());
1450 m_context
->SetClip(&rgn
,CombineModeIntersect
);
1453 void wxGDIPlusContext::Clip( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1455 m_context
->SetClip(RectF(x
,y
,w
,h
),CombineModeIntersect
);
1458 void wxGDIPlusContext::ResetClip()
1460 m_context
->ResetClip();
1463 void wxGDIPlusContext::DrawRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1465 if (m_composition
== wxCOMPOSITION_DEST
)
1468 wxGDIPlusOffsetHelper
helper( m_context
, ShouldOffset() );
1469 Brush
*brush
= m_brush
.IsNull() ? NULL
: ((wxGDIPlusBrushData
*)m_brush
.GetRefData())->GetGDIPlusBrush();
1470 Pen
*pen
= m_pen
.IsNull() ? NULL
: ((wxGDIPlusPenData
*)m_pen
.GetGraphicsData())->GetGDIPlusPen();
1474 // the offset is used to fill only the inside of the rectangle and not paint underneath
1475 // its border which may influence a transparent Pen
1478 offset
= pen
->GetWidth();
1479 m_context
->FillRectangle( brush
, (REAL
)x
+ offset
/2, (REAL
)y
+ offset
/2, (REAL
)w
- offset
, (REAL
)h
- offset
);
1484 m_context
->DrawRectangle( pen
, (REAL
)x
, (REAL
)y
, (REAL
)w
, (REAL
)h
);
1488 void wxGDIPlusContext::StrokeLines( size_t n
, const wxPoint2DDouble
*points
)
1490 if (m_composition
== wxCOMPOSITION_DEST
)
1493 if ( !m_pen
.IsNull() )
1495 wxGDIPlusOffsetHelper
helper( m_context
, ShouldOffset() );
1496 Point
*cpoints
= new Point
[n
];
1497 for (size_t i
= 0; i
< n
; i
++)
1499 cpoints
[i
].X
= (int)(points
[i
].m_x
);
1500 cpoints
[i
].Y
= (int)(points
[i
].m_y
);
1502 } // for (size_t i = 0; i < n; i++)
1503 m_context
->DrawLines( ((wxGDIPlusPenData
*)m_pen
.GetGraphicsData())->GetGDIPlusPen() , cpoints
, n
) ;
1508 void wxGDIPlusContext::DrawLines( size_t n
, const wxPoint2DDouble
*points
, wxPolygonFillMode
WXUNUSED(fillStyle
) )
1510 if (m_composition
== wxCOMPOSITION_DEST
)
1513 wxGDIPlusOffsetHelper
helper( m_context
, ShouldOffset() );
1514 Point
*cpoints
= new Point
[n
];
1515 for (size_t i
= 0; i
< n
; i
++)
1517 cpoints
[i
].X
= (int)(points
[i
].m_x
);
1518 cpoints
[i
].Y
= (int)(points
[i
].m_y
);
1520 } // for (int i = 0; i < n; i++)
1521 if ( !m_brush
.IsNull() )
1522 m_context
->FillPolygon( ((wxGDIPlusBrushData
*)m_brush
.GetRefData())->GetGDIPlusBrush() , cpoints
, n
) ;
1523 if ( !m_pen
.IsNull() )
1524 m_context
->DrawLines( ((wxGDIPlusPenData
*)m_pen
.GetGraphicsData())->GetGDIPlusPen() , cpoints
, n
) ;
1528 void wxGDIPlusContext::StrokePath( const wxGraphicsPath
& path
)
1530 if (m_composition
== wxCOMPOSITION_DEST
)
1533 if ( !m_pen
.IsNull() )
1535 wxGDIPlusOffsetHelper
helper( m_context
, ShouldOffset() );
1536 m_context
->DrawPath( ((wxGDIPlusPenData
*)m_pen
.GetGraphicsData())->GetGDIPlusPen() , (GraphicsPath
*) path
.GetNativePath() );
1540 void wxGDIPlusContext::FillPath( const wxGraphicsPath
& path
, wxPolygonFillMode fillStyle
)
1542 if (m_composition
== wxCOMPOSITION_DEST
)
1545 if ( !m_brush
.IsNull() )
1547 wxGDIPlusOffsetHelper
helper( m_context
, ShouldOffset() );
1548 ((GraphicsPath
*) path
.GetNativePath())->SetFillMode( fillStyle
== wxODDEVEN_RULE
? FillModeAlternate
: FillModeWinding
);
1549 m_context
->FillPath( ((wxGDIPlusBrushData
*)m_brush
.GetRefData())->GetGDIPlusBrush() ,
1550 (GraphicsPath
*) path
.GetNativePath());
1554 bool wxGDIPlusContext::SetAntialiasMode(wxAntialiasMode antialias
)
1556 if (m_antialias
== antialias
)
1559 m_antialias
= antialias
;
1561 SmoothingMode antialiasMode
;
1564 case wxANTIALIAS_DEFAULT
:
1565 antialiasMode
= SmoothingModeHighQuality
;
1567 case wxANTIALIAS_NONE
:
1568 antialiasMode
= SmoothingModeNone
;
1573 m_context
->SetSmoothingMode(antialiasMode
);
1577 bool wxGDIPlusContext::SetInterpolationQuality(wxInterpolationQuality interpolation
)
1579 if (m_interpolation
== interpolation
)
1582 InterpolationMode interpolationMode
= InterpolationModeDefault
;
1583 switch (interpolation
)
1585 case wxINTERPOLATION_DEFAULT
:
1586 interpolationMode
= InterpolationModeDefault
;
1589 case wxINTERPOLATION_NONE
:
1590 interpolationMode
= InterpolationModeNearestNeighbor
;
1593 case wxINTERPOLATION_FAST
:
1594 interpolationMode
= InterpolationModeLowQuality
;
1597 case wxINTERPOLATION_GOOD
:
1598 interpolationMode
= InterpolationModeHighQuality
;
1601 case wxINTERPOLATION_BEST
:
1602 interpolationMode
= InterpolationModeHighQualityBicubic
;
1609 if ( m_context
->SetInterpolationMode(interpolationMode
) != Gdiplus::Ok
)
1612 m_interpolation
= interpolation
;
1617 bool wxGDIPlusContext::SetCompositionMode(wxCompositionMode op
)
1619 if ( m_composition
== op
)
1624 if (m_composition
== wxCOMPOSITION_DEST
)
1627 CompositingMode cop
;
1630 case wxCOMPOSITION_SOURCE
:
1631 cop
= CompositingModeSourceCopy
;
1633 case wxCOMPOSITION_OVER
:
1634 cop
= CompositingModeSourceOver
;
1640 m_context
->SetCompositingMode(cop
);
1644 void wxGDIPlusContext::BeginLayer(wxDouble
/* opacity */)
1649 void wxGDIPlusContext::EndLayer()
1654 void wxGDIPlusContext::Rotate( wxDouble angle
)
1656 m_context
->RotateTransform( RadToDeg(angle
) );
1659 void wxGDIPlusContext::Translate( wxDouble dx
, wxDouble dy
)
1661 m_context
->TranslateTransform( dx
, dy
);
1664 void wxGDIPlusContext::Scale( wxDouble xScale
, wxDouble yScale
)
1666 m_context
->ScaleTransform(xScale
,yScale
);
1669 void wxGDIPlusContext::PushState()
1671 GraphicsState state
= m_context
->Save();
1672 m_stateStack
.push(state
);
1675 void wxGDIPlusContext::PopState()
1677 wxCHECK_RET( !m_stateStack
.empty(), wxT("No state to pop") );
1679 GraphicsState state
= m_stateStack
.top();
1681 m_context
->Restore(state
);
1684 void wxGDIPlusContext::DrawBitmap( const wxGraphicsBitmap
&bmp
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1686 if (m_composition
== wxCOMPOSITION_DEST
)
1689 Bitmap
* image
= static_cast<wxGDIPlusBitmapData
*>(bmp
.GetRefData())->GetGDIPlusBitmap();
1692 if( image
->GetWidth() != (UINT
) w
|| image
->GetHeight() != (UINT
) h
)
1694 Rect
drawRect((REAL
) x
, (REAL
)y
, (REAL
)w
, (REAL
)h
);
1695 m_context
->SetPixelOffsetMode( PixelOffsetModeNone
);
1696 m_context
->DrawImage(image
, drawRect
, 0 , 0 , image
->GetWidth(), image
->GetHeight(), UnitPixel
) ;
1697 m_context
->SetPixelOffsetMode( PixelOffsetModeHalf
);
1700 m_context
->DrawImage(image
,(REAL
) x
,(REAL
) y
,(REAL
) w
,(REAL
) h
) ;
1704 void wxGDIPlusContext::DrawBitmap( const wxBitmap
&bmp
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1706 wxGraphicsBitmap bitmap
= GetRenderer()->CreateBitmap(bmp
);
1707 DrawBitmap(bitmap
, x
, y
, w
, h
);
1710 void wxGDIPlusContext::DrawIcon( const wxIcon
&icon
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
1712 if (m_composition
== wxCOMPOSITION_DEST
)
1715 // the built-in conversion fails when there is alpha in the HICON (eg XP style icons), we can only
1716 // find out by looking at the bitmap data whether there really was alpha in it
1717 HICON hIcon
= (HICON
)icon
.GetHICON();
1719 // IconInfo creates the bitmaps for color and mask, we must dispose of them after use
1720 if (!GetIconInfo(hIcon
,&iconInfo
))
1723 Bitmap
interim(iconInfo
.hbmColor
,NULL
);
1725 Bitmap
* image
= NULL
;
1727 // if it's not 32 bit, it doesn't have an alpha channel, note that since the conversion doesn't
1728 // work correctly, asking IsAlphaPixelFormat at this point fails as well
1729 if( GetPixelFormatSize(interim
.GetPixelFormat())!= 32 )
1731 image
= Bitmap::FromHICON(hIcon
);
1735 size_t width
= interim
.GetWidth();
1736 size_t height
= interim
.GetHeight();
1737 Rect
bounds(0,0,width
,height
);
1740 interim
.LockBits(&bounds
, ImageLockModeRead
,
1741 interim
.GetPixelFormat(),&data
);
1743 bool hasAlpha
= false;
1744 for ( size_t y
= 0 ; y
< height
&& !hasAlpha
; ++y
)
1746 for( size_t x
= 0 ; x
< width
&& !hasAlpha
; ++x
)
1748 ARGB
*dest
= (ARGB
*)((BYTE
*)data
.Scan0
+ data
.Stride
*y
+ x
*4);
1749 if ( ( *dest
& Color::AlphaMask
) != 0 )
1756 image
= new Bitmap(data
.Width
, data
.Height
, data
.Stride
,
1757 PixelFormat32bppARGB
, (BYTE
*) data
.Scan0
);
1761 image
= Bitmap::FromHICON(hIcon
);
1764 interim
.UnlockBits(&data
);
1767 m_context
->DrawImage(image
,(REAL
) x
,(REAL
) y
,(REAL
) w
,(REAL
) h
) ;
1770 DeleteObject(iconInfo
.hbmColor
);
1771 DeleteObject(iconInfo
.hbmMask
);
1774 void wxGDIPlusContext::DoDrawText(const wxString
& str
,
1775 wxDouble x
, wxDouble y
)
1777 if (m_composition
== wxCOMPOSITION_DEST
)
1780 wxCHECK_RET( !m_font
.IsNull(),
1781 wxT("wxGDIPlusContext::DrawText - no valid font set") );
1786 wxGDIPlusFontData
* const
1787 fontData
= (wxGDIPlusFontData
*)m_font
.GetRefData();
1789 m_context
->DrawString
1791 str
.wc_str(*wxConvUI
), // string to draw, always Unicode
1792 -1, // length: string is NUL-terminated
1793 fontData
->GetGDIPlusFont(),
1795 GetDrawTextStringFormat(),
1796 fontData
->GetGDIPlusBrush()
1800 void wxGDIPlusContext::GetTextExtent( const wxString
&str
, wxDouble
*width
, wxDouble
*height
,
1801 wxDouble
*descent
, wxDouble
*externalLeading
) const
1803 wxCHECK_RET( !m_font
.IsNull(), wxT("wxGDIPlusContext::GetTextExtent - no valid font set") );
1805 wxWCharBuffer s
= str
.wc_str( *wxConvUI
);
1806 FontFamily ffamily
;
1807 Font
* f
= ((wxGDIPlusFontData
*)m_font
.GetRefData())->GetGDIPlusFont();
1809 f
->GetFamily(&ffamily
) ;
1811 REAL factorY
= m_fontScaleRatio
;
1813 // Notice that we must use the real font style or the results would be
1814 // incorrect for italic/bold fonts.
1815 const INT style
= f
->GetStyle();
1816 const REAL size
= f
->GetSize();
1817 const REAL emHeight
= ffamily
.GetEmHeight(style
);
1818 REAL rDescent
= ffamily
.GetCellDescent(style
) * size
/ emHeight
;
1819 REAL rAscent
= ffamily
.GetCellAscent(style
) * size
/ emHeight
;
1820 REAL rHeight
= ffamily
.GetLineSpacing(style
) * size
/ emHeight
;
1823 *height
= rHeight
* factorY
;
1825 *descent
= rDescent
* factorY
;
1826 if ( externalLeading
)
1827 *externalLeading
= (rHeight
- rAscent
- rDescent
) * factorY
;
1828 // measuring empty strings is not guaranteed, so do it by hand
1836 RectF
layoutRect(0,0, 100000.0f
, 100000.0f
);
1839 m_context
->MeasureString((const wchar_t *) s
, wcslen(s
) , f
, layoutRect
, GetDrawTextStringFormat(), &bounds
) ;
1841 *width
= bounds
.Width
;
1843 *height
= bounds
.Height
;
1847 void wxGDIPlusContext::GetPartialTextExtents(const wxString
& text
, wxArrayDouble
& widths
) const
1850 widths
.Add(0, text
.length());
1852 wxCHECK_RET( !m_font
.IsNull(), wxT("wxGDIPlusContext::GetPartialTextExtents - no valid font set") );
1857 Font
* f
= ((wxGDIPlusFontData
*)m_font
.GetRefData())->GetGDIPlusFont();
1858 wxWCharBuffer ws
= text
.wc_str( *wxConvUI
);
1859 size_t len
= wcslen( ws
) ;
1860 wxASSERT_MSG(text
.length() == len
, wxT("GetPartialTextExtents not yet implemented for multichar situations"));
1862 RectF
layoutRect(0,0, 100000.0f
, 100000.0f
);
1863 StringFormat
strFormat( GetDrawTextStringFormat() );
1865 size_t startPosition
= 0;
1866 size_t remainder
= len
;
1867 const size_t maxSpan
= 32;
1868 CharacterRange
* ranges
= new CharacterRange
[maxSpan
] ;
1869 Region
* regions
= new Region
[maxSpan
];
1871 while( remainder
> 0 )
1873 size_t span
= wxMin( maxSpan
, remainder
);
1875 for( size_t i
= 0 ; i
< span
; ++i
)
1877 ranges
[i
].First
= 0 ;
1878 ranges
[i
].Length
= startPosition
+i
+1 ;
1880 strFormat
.SetMeasurableCharacterRanges(span
,ranges
);
1881 m_context
->MeasureCharacterRanges(ws
, -1 , f
,layoutRect
, &strFormat
,span
,regions
) ;
1884 for ( size_t i
= 0 ; i
< span
; ++i
)
1886 regions
[i
].GetBounds(&bbox
,m_context
);
1887 widths
[startPosition
+i
] = bbox
.Width
;
1890 startPosition
+= span
;
1897 bool wxGDIPlusContext::ShouldOffset() const
1899 if ( !m_enableOffset
)
1903 if ( !m_pen
.IsNull() )
1905 penwidth
= (int)((wxGDIPlusPenData
*)m_pen
.GetRefData())->GetWidth();
1906 if ( penwidth
== 0 )
1909 return ( penwidth
% 2 ) == 1;
1912 void* wxGDIPlusContext::GetNativeContext()
1917 // concatenates this transform with the current transform of this context
1918 void wxGDIPlusContext::ConcatTransform( const wxGraphicsMatrix
& matrix
)
1920 m_context
->MultiplyTransform((Matrix
*) matrix
.GetNativeMatrix());
1923 // sets the transform of this context
1924 void wxGDIPlusContext::SetTransform( const wxGraphicsMatrix
& matrix
)
1926 m_context
->SetTransform((Matrix
*) matrix
.GetNativeMatrix());
1929 // gets the matrix of this context
1930 wxGraphicsMatrix
wxGDIPlusContext::GetTransform() const
1932 wxGraphicsMatrix matrix
= CreateMatrix();
1933 m_context
->GetTransform((Matrix
*) matrix
.GetNativeMatrix());
1937 void wxGDIPlusContext::GetSize( wxDouble
* width
, wxDouble
*height
)
1943 //-----------------------------------------------------------------------------
1944 // wxGDIPlusPrintingContext implementation
1945 //-----------------------------------------------------------------------------
1947 wxGDIPlusPrintingContext::wxGDIPlusPrintingContext( wxGraphicsRenderer
* renderer
,
1949 : wxGDIPlusContext(renderer
, dc
)
1951 Graphics
* context
= GetGraphics();
1953 //m_context->SetPageUnit(UnitDocument);
1955 // Setup page scale, based on DPI ratio.
1956 // Antecedent should be 100dpi when the default page unit
1957 // (UnitDisplay) is used. Page unit UnitDocument would require 300dpi
1958 // instead. Note that calling SetPageScale() does not have effect on
1959 // non-printing DCs (that is, any other than wxPrinterDC or
1960 // wxEnhMetaFileDC).
1961 REAL dpiRatio
= 100.0 / context
->GetDpiY();
1962 context
->SetPageScale(dpiRatio
);
1964 // We use this modifier when measuring fonts. It is needed because the
1965 // page scale is modified above.
1966 m_fontScaleRatio
= context
->GetDpiY() / 72.0;
1969 //-----------------------------------------------------------------------------
1970 // wxGDIPlusRenderer implementation
1971 //-----------------------------------------------------------------------------
1973 IMPLEMENT_DYNAMIC_CLASS(wxGDIPlusRenderer
,wxGraphicsRenderer
)
1975 static wxGDIPlusRenderer gs_GDIPlusRenderer
;
1977 wxGraphicsRenderer
* wxGraphicsRenderer::GetDefaultRenderer()
1979 return &gs_GDIPlusRenderer
;
1982 bool wxGDIPlusRenderer::EnsureIsLoaded()
1984 // load gdiplus.dll if not yet loaded, but don't bother doing it again
1985 // if we already tried and failed (we don't want to spend lot of time
1986 // returning NULL from wxGraphicsContext::Create(), which may be called
1987 // relatively frequently):
1988 if ( m_loaded
== -1 )
1993 return m_loaded
== 1;
1996 // call EnsureIsLoaded() and return returnOnFail value if it fails
1997 #define ENSURE_LOADED_OR_RETURN(returnOnFail) \
1998 if ( !EnsureIsLoaded() ) \
1999 return (returnOnFail)
2002 void wxGDIPlusRenderer::Load()
2004 GdiplusStartupInput input
;
2005 GdiplusStartupOutput output
;
2006 if ( GdiplusStartup(&m_gditoken
,&input
,&output
) == Gdiplus::Ok
)
2008 wxLogTrace("gdiplus", "successfully initialized GDI+");
2013 wxLogTrace("gdiplus", "failed to initialize GDI+, missing gdiplus.dll?");
2018 void wxGDIPlusRenderer::Unload()
2022 GdiplusShutdown(m_gditoken
);
2025 m_loaded
= -1; // next Load() will try again
2028 wxGraphicsContext
* wxGDIPlusRenderer::CreateContext( const wxWindowDC
& dc
)
2030 ENSURE_LOADED_OR_RETURN(NULL
);
2031 wxGDIPlusContext
* context
= new wxGDIPlusContext(this, dc
);
2032 context
->EnableOffset(true);
2036 #if wxUSE_PRINTING_ARCHITECTURE
2037 wxGraphicsContext
* wxGDIPlusRenderer::CreateContext( const wxPrinterDC
& dc
)
2039 ENSURE_LOADED_OR_RETURN(NULL
);
2040 wxGDIPlusContext
* context
= new wxGDIPlusPrintingContext(this, dc
);
2045 #if wxUSE_ENH_METAFILE
2046 wxGraphicsContext
* wxGDIPlusRenderer::CreateContext( const wxEnhMetaFileDC
& dc
)
2048 ENSURE_LOADED_OR_RETURN(NULL
);
2049 wxGDIPlusContext
* context
= new wxGDIPlusPrintingContext(this, dc
);
2054 wxGraphicsContext
* wxGDIPlusRenderer::CreateContext( const wxMemoryDC
& dc
)
2056 ENSURE_LOADED_OR_RETURN(NULL
);
2057 wxGDIPlusContext
* context
= new wxGDIPlusContext(this, dc
);
2058 context
->EnableOffset(true);
2063 wxGraphicsContext
* wxGDIPlusRenderer::CreateContextFromImage(wxImage
& image
)
2065 ENSURE_LOADED_OR_RETURN(NULL
);
2066 wxGDIPlusContext
* context
= new wxGDIPlusImageContext(this, image
);
2067 context
->EnableOffset(true);
2071 #endif // wxUSE_IMAGE
2073 wxGraphicsContext
* wxGDIPlusRenderer::CreateMeasuringContext()
2075 ENSURE_LOADED_OR_RETURN(NULL
);
2076 return new wxGDIPlusMeasuringContext(this);
2079 wxGraphicsContext
* wxGDIPlusRenderer::CreateContextFromNativeContext( void * context
)
2081 ENSURE_LOADED_OR_RETURN(NULL
);
2082 return new wxGDIPlusContext(this,(Graphics
*) context
);
2086 wxGraphicsContext
* wxGDIPlusRenderer::CreateContextFromNativeWindow( void * window
)
2088 ENSURE_LOADED_OR_RETURN(NULL
);
2089 return new wxGDIPlusContext(this,(HWND
) window
);
2092 wxGraphicsContext
* wxGDIPlusRenderer::CreateContext( wxWindow
* window
)
2094 ENSURE_LOADED_OR_RETURN(NULL
);
2095 return new wxGDIPlusContext(this, (HWND
) window
->GetHWND() );
2100 wxGraphicsPath
wxGDIPlusRenderer::CreatePath()
2102 ENSURE_LOADED_OR_RETURN(wxNullGraphicsPath
);
2104 m
.SetRefData( new wxGDIPlusPathData(this));
2111 wxGraphicsMatrix
wxGDIPlusRenderer::CreateMatrix( wxDouble a
, wxDouble b
, wxDouble c
, wxDouble d
,
2112 wxDouble tx
, wxDouble ty
)
2115 ENSURE_LOADED_OR_RETURN(wxNullGraphicsMatrix
);
2117 wxGDIPlusMatrixData
* data
= new wxGDIPlusMatrixData( this );
2118 data
->Set( a
,b
,c
,d
,tx
,ty
) ;
2123 wxGraphicsPen
wxGDIPlusRenderer::CreatePen(const wxPen
& pen
)
2125 ENSURE_LOADED_OR_RETURN(wxNullGraphicsPen
);
2126 if ( !pen
.IsOk() || pen
.GetStyle() == wxTRANSPARENT
)
2127 return wxNullGraphicsPen
;
2131 p
.SetRefData(new wxGDIPlusPenData( this, pen
));
2136 wxGraphicsBrush
wxGDIPlusRenderer::CreateBrush(const wxBrush
& brush
)
2138 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBrush
);
2139 if ( !brush
.IsOk() || brush
.GetStyle() == wxTRANSPARENT
)
2140 return wxNullGraphicsBrush
;
2144 p
.SetRefData(new wxGDIPlusBrushData( this, brush
));
2150 wxGDIPlusRenderer::CreateLinearGradientBrush(wxDouble x1
, wxDouble y1
,
2151 wxDouble x2
, wxDouble y2
,
2152 const wxGraphicsGradientStops
& stops
)
2154 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBrush
);
2156 wxGDIPlusBrushData
* d
= new wxGDIPlusBrushData( this );
2157 d
->CreateLinearGradientBrush(x1
, y1
, x2
, y2
, stops
);
2163 wxGDIPlusRenderer::CreateRadialGradientBrush(wxDouble xo
, wxDouble yo
,
2164 wxDouble xc
, wxDouble yc
,
2166 const wxGraphicsGradientStops
& stops
)
2168 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBrush
);
2170 wxGDIPlusBrushData
* d
= new wxGDIPlusBrushData( this );
2171 d
->CreateRadialGradientBrush(xo
,yo
,xc
,yc
,radius
,stops
);
2177 wxGDIPlusRenderer::CreateFont( const wxFont
&font
,
2178 const wxColour
&col
)
2180 ENSURE_LOADED_OR_RETURN(wxNullGraphicsFont
);
2184 p
.SetRefData(new wxGDIPlusFontData( this, font
, col
));
2188 return wxNullGraphicsFont
;
2192 wxGDIPlusRenderer::CreateFont(double size
,
2193 const wxString
& facename
,
2195 const wxColour
& col
)
2197 ENSURE_LOADED_OR_RETURN(wxNullGraphicsFont
);
2199 // Convert wxFont flags to GDI+ style:
2200 int style
= FontStyleRegular
;
2201 if ( flags
& wxFONTFLAG_ITALIC
)
2202 style
|= FontStyleItalic
;
2203 if ( flags
& wxFONTFLAG_UNDERLINED
)
2204 style
|= FontStyleUnderline
;
2205 if ( flags
& wxFONTFLAG_BOLD
)
2206 style
|= FontStyleBold
;
2207 if ( flags
& wxFONTFLAG_STRIKETHROUGH
)
2208 style
|= FontStyleStrikeout
;
2212 f
.SetRefData(new wxGDIPlusFontData(this, facename
, size
, style
, col
));
2216 wxGraphicsBitmap
wxGDIPlusRenderer::CreateBitmap( const wxBitmap
&bitmap
)
2218 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBitmap
);
2219 if ( bitmap
.IsOk() )
2222 p
.SetRefData(new wxGDIPlusBitmapData( this , bitmap
));
2226 return wxNullGraphicsBitmap
;
2231 wxGraphicsBitmap
wxGDIPlusRenderer::CreateBitmapFromImage(const wxImage
& image
)
2233 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBitmap
);
2236 // Notice that we rely on conversion from wxImage to wxBitmap here but
2237 // we could probably do it more efficiently by converting from wxImage
2238 // to GDI+ Bitmap directly, i.e. copying wxImage pixels to the buffer
2239 // returned by Bitmap::LockBits(). However this would require writing
2240 // code specific for this task while like this we can reuse existing
2241 // code (see also wxGDIPlusBitmapData::ConvertToImage()).
2242 wxGraphicsBitmap gb
;
2243 gb
.SetRefData(new wxGDIPlusBitmapData(this, image
));
2247 return wxNullGraphicsBitmap
;
2251 wxImage
wxGDIPlusRenderer::CreateImageFromBitmap(const wxGraphicsBitmap
& bmp
)
2253 ENSURE_LOADED_OR_RETURN(wxNullImage
);
2254 const wxGDIPlusBitmapData
* const
2255 data
= static_cast<wxGDIPlusBitmapData
*>(bmp
.GetGraphicsData());
2257 return data
? data
->ConvertToImage() : wxNullImage
;
2260 #endif // wxUSE_IMAGE
2263 wxGraphicsBitmap
wxGDIPlusRenderer::CreateBitmapFromNativeBitmap( void *bitmap
)
2265 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBitmap
);
2266 if ( bitmap
!= NULL
)
2269 p
.SetRefData(new wxGDIPlusBitmapData( this , (Bitmap
*) bitmap
));
2273 return wxNullGraphicsBitmap
;
2276 wxGraphicsBitmap
wxGDIPlusRenderer::CreateSubBitmap( const wxGraphicsBitmap
&bitmap
, wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
2278 ENSURE_LOADED_OR_RETURN(wxNullGraphicsBitmap
);
2279 Bitmap
* image
= static_cast<wxGDIPlusBitmapData
*>(bitmap
.GetRefData())->GetGDIPlusBitmap();
2283 p
.SetRefData(new wxGDIPlusBitmapData( this , image
->Clone( (REAL
) x
, (REAL
) y
, (REAL
) w
, (REAL
) h
, PixelFormat32bppPARGB
) ));
2287 return wxNullGraphicsBitmap
;
2290 // Shutdown GDI+ at app exit, before possible dll unload
2291 class wxGDIPlusRendererModule
: public wxModule
2294 virtual bool OnInit() { return true; }
2295 virtual void OnExit()
2297 wxDELETE(gs_drawTextStringFormat
);
2299 gs_GDIPlusRenderer
.Unload();
2303 DECLARE_DYNAMIC_CLASS(wxGDIPlusRendererModule
)
2306 IMPLEMENT_DYNAMIC_CLASS(wxGDIPlusRendererModule
, wxModule
)
2308 // ----------------------------------------------------------------------------
2309 // wxMSW-specific parts of wxGCDC
2310 // ----------------------------------------------------------------------------
2312 WXHDC
wxGCDC::AcquireHDC()
2314 wxGraphicsContext
* const gc
= GetGraphicsContext();
2319 // we can't get the HDC if it is not a GDI+ context
2320 wxGraphicsRenderer
* r1
= gc
->GetRenderer();
2321 wxGraphicsRenderer
* r2
= wxGraphicsRenderer::GetCairoRenderer();
2326 Graphics
* const g
= static_cast<Graphics
*>(gc
->GetNativeContext());
2327 return g
? g
->GetHDC() : NULL
;
2330 void wxGCDC::ReleaseHDC(WXHDC hdc
)
2335 wxGraphicsContext
* const gc
= GetGraphicsContext();
2336 wxCHECK_RET( gc
, "can't release HDC because there is no wxGraphicsContext" );
2339 // we can't get the HDC if it is not a GDI+ context
2340 wxGraphicsRenderer
* r1
= gc
->GetRenderer();
2341 wxGraphicsRenderer
* r2
= wxGraphicsRenderer::GetCairoRenderer();
2346 Graphics
* const g
= static_cast<Graphics
*>(gc
->GetNativeContext());
2347 wxCHECK_RET( g
, "can't release HDC because there is no Graphics" );
2349 g
->ReleaseHDC((HDC
)hdc
);
2352 #endif // wxUSE_GRAPHICS_CONTEXT