]>
git.saurik.com Git - wxWidgets.git/blob - src/common/graphcmn.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/graphcmn.cpp
3 // Purpose: graphics context methods common to all platforms
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // For compilers that support precompilation, includes "wx.h".
13 #include "wx/wxprec.h"
15 #if defined(__BORLANDC__)
19 #if wxUSE_GRAPHICS_CONTEXT
21 #include "wx/graphics.h"
25 #include "wx/bitmap.h"
26 #include "wx/dcmemory.h"
27 #include "wx/region.h"
31 #if !defined(wxMAC_USE_CORE_GRAPHICS_BLEND_MODES)
32 #define wxMAC_USE_CORE_GRAPHICS_BLEND_MODES 0
35 //-----------------------------------------------------------------------------
37 //-----------------------------------------------------------------------------
39 static const double RAD2DEG
= 180.0 / M_PI
;
41 //-----------------------------------------------------------------------------
43 //-----------------------------------------------------------------------------
45 static inline double DegToRad(double deg
)
47 return (deg
* M_PI
) / 180.0;
50 //-----------------------------------------------------------------------------
52 //-----------------------------------------------------------------------------
54 //-----------------------------------------------------------------------------
56 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsObject
, wxObject
)
58 wxGraphicsObjectRefData::wxGraphicsObjectRefData( wxGraphicsRenderer
* renderer
)
60 m_renderer
= renderer
;
62 wxGraphicsObjectRefData::wxGraphicsObjectRefData( const wxGraphicsObjectRefData
* data
)
64 m_renderer
= data
->m_renderer
;
66 wxGraphicsRenderer
* wxGraphicsObjectRefData::GetRenderer() const
71 wxGraphicsObjectRefData
* wxGraphicsObjectRefData::Clone() const
73 return new wxGraphicsObjectRefData(this);
76 wxGraphicsObject::wxGraphicsObject()
80 wxGraphicsObject::wxGraphicsObject( wxGraphicsRenderer
* renderer
)
82 SetRefData( new wxGraphicsObjectRefData(renderer
));
85 wxGraphicsObject::~wxGraphicsObject()
89 bool wxGraphicsObject::IsNull() const
91 return m_refData
== NULL
;
94 wxGraphicsRenderer
* wxGraphicsObject::GetRenderer() const
96 return ( IsNull() ? NULL
: GetGraphicsData()->GetRenderer() );
99 wxGraphicsObjectRefData
* wxGraphicsObject::GetGraphicsData() const
101 return (wxGraphicsObjectRefData
*) m_refData
;
104 wxObjectRefData
* wxGraphicsObject::CreateRefData() const
106 wxLogDebug(wxT("A Null Object cannot be changed"));
110 wxObjectRefData
* wxGraphicsObject::CloneRefData(const wxObjectRefData
* data
) const
112 const wxGraphicsObjectRefData
* ptr
= (const wxGraphicsObjectRefData
*) data
;
116 //-----------------------------------------------------------------------------
118 //-----------------------------------------------------------------------------
120 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsPen
, wxGraphicsObject
)
121 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsBrush
, wxGraphicsObject
)
122 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsFont
, wxGraphicsObject
)
124 WXDLLIMPEXP_DATA_CORE(wxGraphicsPen
) wxNullGraphicsPen
;
125 WXDLLIMPEXP_DATA_CORE(wxGraphicsBrush
) wxNullGraphicsBrush
;
126 WXDLLIMPEXP_DATA_CORE(wxGraphicsFont
) wxNullGraphicsFont
;
128 //-----------------------------------------------------------------------------
130 //-----------------------------------------------------------------------------
132 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsMatrix
, wxGraphicsObject
)
133 WXDLLIMPEXP_DATA_CORE(wxGraphicsMatrix
) wxNullGraphicsMatrix
;
135 // concatenates the matrix
136 void wxGraphicsMatrix::Concat( const wxGraphicsMatrix
*t
)
139 GetMatrixData()->Concat(t
->GetMatrixData());
142 // sets the matrix to the respective values
143 void wxGraphicsMatrix::Set(wxDouble a
, wxDouble b
, wxDouble c
, wxDouble d
,
144 wxDouble tx
, wxDouble ty
)
147 GetMatrixData()->Set(a
,b
,c
,d
,tx
,ty
);
150 // makes this the inverse matrix
151 void wxGraphicsMatrix::Invert()
154 GetMatrixData()->Invert();
157 // returns true if the elements of the transformation matrix are equal ?
158 bool wxGraphicsMatrix::IsEqual( const wxGraphicsMatrix
* t
) const
160 return GetMatrixData()->IsEqual(t
->GetMatrixData());
163 // return true if this is the identity matrix
164 bool wxGraphicsMatrix::IsIdentity() const
166 return GetMatrixData()->IsIdentity();
169 // add the translation to this matrix
170 void wxGraphicsMatrix::Translate( wxDouble dx
, wxDouble dy
)
173 GetMatrixData()->Translate(dx
,dy
);
176 // add the scale to this matrix
177 void wxGraphicsMatrix::Scale( wxDouble xScale
, wxDouble yScale
)
180 GetMatrixData()->Scale(xScale
,yScale
);
183 // add the rotation to this matrix (radians)
184 void wxGraphicsMatrix::Rotate( wxDouble angle
)
187 GetMatrixData()->Rotate(angle
);
191 // apply the transforms
194 // applies that matrix to the point
195 void wxGraphicsMatrix::TransformPoint( wxDouble
*x
, wxDouble
*y
) const
197 GetMatrixData()->TransformPoint(x
,y
);
200 // applies the matrix except for translations
201 void wxGraphicsMatrix::TransformDistance( wxDouble
*dx
, wxDouble
*dy
) const
203 GetMatrixData()->TransformDistance(dx
,dy
);
206 // returns the native representation
207 void * wxGraphicsMatrix::GetNativeMatrix() const
209 return GetMatrixData()->GetNativeMatrix();
212 //-----------------------------------------------------------------------------
214 //-----------------------------------------------------------------------------
216 IMPLEMENT_DYNAMIC_CLASS(wxGraphicsPath
, wxGraphicsObject
)
218 // convenience functions, for using wxPoint2DDouble etc
220 wxPoint2DDouble
wxGraphicsPath::GetCurrentPoint() const
223 GetCurrentPoint(&x
,&y
);
224 return wxPoint2DDouble(x
,y
);
227 void wxGraphicsPath::MoveToPoint( const wxPoint2DDouble
& p
)
229 MoveToPoint( p
.m_x
, p
.m_y
);
232 void wxGraphicsPath::AddLineToPoint( const wxPoint2DDouble
& p
)
234 AddLineToPoint( p
.m_x
, p
.m_y
);
237 void wxGraphicsPath::AddCurveToPoint( const wxPoint2DDouble
& c1
, const wxPoint2DDouble
& c2
, const wxPoint2DDouble
& e
)
239 AddCurveToPoint(c1
.m_x
, c1
.m_y
, c2
.m_x
, c2
.m_y
, e
.m_x
, e
.m_y
);
242 void wxGraphicsPath::AddArc( const wxPoint2DDouble
& c
, wxDouble r
, wxDouble startAngle
, wxDouble endAngle
, bool clockwise
)
244 AddArc(c
.m_x
, c
.m_y
, r
, startAngle
, endAngle
, clockwise
);
247 wxRect2DDouble
wxGraphicsPath::GetBox() const
251 return wxRect2DDouble( x
,y
,w
,h
);
254 bool wxGraphicsPath::Contains( const wxPoint2DDouble
& c
, int fillStyle
) const
256 return Contains( c
.m_x
, c
.m_y
, fillStyle
);
261 // begins a new subpath at (x,y)
262 void wxGraphicsPath::MoveToPoint( wxDouble x
, wxDouble y
)
265 GetPathData()->MoveToPoint(x
,y
);
268 // adds a straight line from the current point to (x,y)
269 void wxGraphicsPath::AddLineToPoint( wxDouble x
, wxDouble y
)
272 GetPathData()->AddLineToPoint(x
,y
);
275 // adds a cubic Bezier curve from the current point, using two control points and an end point
276 void wxGraphicsPath::AddCurveToPoint( wxDouble cx1
, wxDouble cy1
, wxDouble cx2
, wxDouble cy2
, wxDouble x
, wxDouble y
)
279 GetPathData()->AddCurveToPoint(cx1
,cy1
,cx2
,cy2
,x
,y
);
283 void wxGraphicsPath::AddPath( const wxGraphicsPath
& path
)
286 GetPathData()->AddPath(path
.GetPathData());
289 // closes the current sub-path
290 void wxGraphicsPath::CloseSubpath()
293 GetPathData()->CloseSubpath();
296 // gets the last point of the current path, (0,0) if not yet set
297 void wxGraphicsPath::GetCurrentPoint( wxDouble
* x
, wxDouble
* y
) const
299 GetPathData()->GetCurrentPoint(x
,y
);
302 // adds an arc of a circle centering at (x,y) with radius (r) from startAngle to endAngle
303 void wxGraphicsPath::AddArc( wxDouble x
, wxDouble y
, wxDouble r
, wxDouble startAngle
, wxDouble endAngle
, bool clockwise
)
306 GetPathData()->AddArc(x
,y
,r
,startAngle
,endAngle
,clockwise
);
310 // These are convenience functions which - if not available natively will be assembled
311 // using the primitives from above
314 // adds a quadratic Bezier curve from the current point, using a control point and an end point
315 void wxGraphicsPath::AddQuadCurveToPoint( wxDouble cx
, wxDouble cy
, wxDouble x
, wxDouble y
)
318 GetPathData()->AddQuadCurveToPoint(cx
,cy
,x
,y
);
321 // appends a rectangle as a new closed subpath
322 void wxGraphicsPath::AddRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
325 GetPathData()->AddRectangle(x
,y
,w
,h
);
328 // appends an ellipsis as a new closed subpath fitting the passed rectangle
329 void wxGraphicsPath::AddCircle( wxDouble x
, wxDouble y
, wxDouble r
)
332 GetPathData()->AddCircle(x
,y
,r
);
335 // appends 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)
336 void wxGraphicsPath::AddArcToPoint( wxDouble x1
, wxDouble y1
, wxDouble x2
, wxDouble y2
, wxDouble r
)
338 GetPathData()->AddArcToPoint(x1
,y1
,x2
,y2
,r
);
341 // appends an ellipse
342 void wxGraphicsPath::AddEllipse( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
345 GetPathData()->AddEllipse(x
,y
,w
,h
);
348 // appends a rounded rectangle
349 void wxGraphicsPath::AddRoundedRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
, wxDouble radius
)
352 GetPathData()->AddRoundedRectangle(x
,y
,w
,h
,radius
);
355 // returns the native path
356 void * wxGraphicsPath::GetNativePath() const
358 return GetPathData()->GetNativePath();
361 // give the native path returned by GetNativePath() back (there might be some deallocations necessary)
362 void wxGraphicsPath::UnGetNativePath(void *p
)const
364 GetPathData()->UnGetNativePath(p
);
367 // transforms each point of this path by the matrix
368 void wxGraphicsPath::Transform( const wxGraphicsMatrix
& matrix
)
371 GetPathData()->Transform(matrix
.GetMatrixData());
374 // gets the bounding box enclosing all points (possibly including control points)
375 void wxGraphicsPath::GetBox(wxDouble
*x
, wxDouble
*y
, wxDouble
*w
, wxDouble
*h
) const
377 GetPathData()->GetBox(x
,y
,w
,h
);
380 bool wxGraphicsPath::Contains( wxDouble x
, wxDouble y
, int fillStyle
) const
382 return GetPathData()->Contains(x
,y
,fillStyle
);
386 // Emulations, these mus be implemented in the ...Data classes in order to allow for proper overrides
389 void wxGraphicsPathData::AddQuadCurveToPoint( wxDouble cx
, wxDouble cy
, wxDouble x
, wxDouble y
)
391 // calculate using degree elevation to a cubic bezier
395 wxPoint2DDouble start
;
396 GetCurrentPoint(&start
.m_x
,&start
.m_y
);
397 wxPoint2DDouble
end(x
,y
);
398 wxPoint2DDouble
c(cx
,cy
);
399 c1
= wxDouble(1/3.0) * start
+ wxDouble(2/3.0) * c
;
400 c2
= wxDouble(2/3.0) * c
+ wxDouble(1/3.0) * end
;
401 AddCurveToPoint(c1
.m_x
,c1
.m_y
,c2
.m_x
,c2
.m_y
,x
,y
);
404 void wxGraphicsPathData::AddRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
407 AddLineToPoint(x
,y
+h
);
408 AddLineToPoint(x
+w
,y
+h
);
409 AddLineToPoint(x
+w
,y
);
413 void wxGraphicsPathData::AddCircle( wxDouble x
, wxDouble y
, wxDouble r
)
416 AddArc( x
,y
,r
,0,2*M_PI
,false);
420 void wxGraphicsPathData::AddEllipse( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
424 wxDouble xc
= x
+ rw
;
425 wxDouble yc
= y
+ rh
;
426 wxGraphicsMatrix m
= GetRenderer()->CreateMatrix();
429 wxGraphicsPath p
= GetRenderer()->CreatePath();
432 AddPath(p
.GetPathData());
435 void wxGraphicsPathData::AddRoundedRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
, wxDouble radius
)
438 AddRectangle(x
,y
,w
,h
);
441 MoveToPoint( x
+ w
, y
+ h
/ 2);
442 AddArcToPoint(x
+ w
, y
+ h
, x
+ w
/ 2, y
+ h
, radius
);
443 AddArcToPoint(x
, y
+ h
, x
, y
+ h
/ 2, radius
);
444 AddArcToPoint(x
, y
, x
+ w
/ 2, y
, radius
);
445 AddArcToPoint(x
+ w
, y
, x
+ w
, y
+ h
/ 2, radius
);
450 // 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)
451 void wxGraphicsPathData::AddArcToPoint( wxDouble x1
, wxDouble y1
, wxDouble x2
, wxDouble y2
, wxDouble r
)
453 wxPoint2DDouble current
;
454 GetCurrentPoint(¤t
.m_x
,¤t
.m_y
);
455 wxPoint2DDouble
p1(x1
,y1
);
456 wxPoint2DDouble
p2(x2
,y2
);
458 wxPoint2DDouble v1
= current
- p1
;
460 wxPoint2DDouble v2
= p2
- p1
;
463 wxDouble alpha
= v1
.GetVectorAngle() - v2
.GetVectorAngle();
467 // TODO obtuse angles
469 alpha
= DegToRad(alpha
);
471 wxDouble dist
= r
/ sin(alpha
/2) * cos(alpha
/2);
472 // calculate tangential points
473 wxPoint2DDouble t1
= dist
*v1
+ p1
;
474 wxPoint2DDouble t2
= dist
*v2
+ p1
;
476 wxPoint2DDouble nv1
= v1
;
477 nv1
.SetVectorAngle(v1
.GetVectorAngle()-90);
478 wxPoint2DDouble c
= t1
+ r
*nv1
;
480 wxDouble a1
= v1
.GetVectorAngle()+90;
481 wxDouble a2
= v2
.GetVectorAngle()-90;
483 AddLineToPoint(t1
.m_x
,t1
.m_y
);
484 AddArc(c
.m_x
,c
.m_y
,r
,DegToRad(a1
),DegToRad(a2
),true);
485 AddLineToPoint(p2
.m_x
,p2
.m_y
);
488 //-----------------------------------------------------------------------------
489 // wxGraphicsContext Convenience Methods
490 //-----------------------------------------------------------------------------
492 IMPLEMENT_ABSTRACT_CLASS(wxGraphicsContext
, wxObject
)
495 wxGraphicsContext::wxGraphicsContext(wxGraphicsRenderer
* renderer
) : wxGraphicsObject(renderer
)
499 wxGraphicsContext::~wxGraphicsContext()
504 void wxGraphicsContext::SetPen( const wxGraphicsPen
& pen
)
509 void wxGraphicsContext::SetPen( const wxPen
& pen
)
511 if ( pen
.GetStyle() == wxTRANSPARENT
)
512 SetPen( wxNullGraphicsPen
);
514 SetPen( CreatePen( pen
) );
517 // sets the brush for filling
518 void wxGraphicsContext::SetBrush( const wxGraphicsBrush
& brush
)
523 void wxGraphicsContext::SetBrush( const wxBrush
& brush
)
525 if ( brush
.GetStyle() == wxTRANSPARENT
)
526 SetBrush( wxNullGraphicsBrush
);
528 SetBrush( CreateBrush( brush
) );
531 // sets the brush for filling
532 void wxGraphicsContext::SetFont( const wxGraphicsFont
& font
)
537 void wxGraphicsContext::SetFont( const wxFont
& font
, const wxColour
& colour
)
540 SetFont( CreateFont( font
, colour
) );
542 SetFont( wxNullGraphicsFont
);
545 void wxGraphicsContext::DrawPath( const wxGraphicsPath
& path
, int fillStyle
)
547 FillPath( path
, fillStyle
);
551 void wxGraphicsContext::DrawText( const wxString
&str
, wxDouble x
, wxDouble y
, wxDouble angle
)
555 DrawText( str
, 0, 0 );
560 void wxGraphicsContext::StrokeLine( wxDouble x1
, wxDouble y1
, wxDouble x2
, wxDouble y2
)
562 wxGraphicsPath path
= CreatePath();
563 path
.MoveToPoint(x1
, y1
);
564 path
.AddLineToPoint( x2
, y2
);
568 void wxGraphicsContext::DrawRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
570 wxGraphicsPath path
= CreatePath();
571 path
.AddRectangle( x
, y
, w
, h
);
575 void wxGraphicsContext::DrawEllipse( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
)
577 wxGraphicsPath path
= CreatePath();
578 path
.AddEllipse(x
,y
,w
,h
);
582 void wxGraphicsContext::DrawRoundedRectangle( wxDouble x
, wxDouble y
, wxDouble w
, wxDouble h
, wxDouble radius
)
584 wxGraphicsPath path
= CreatePath();
585 path
.AddRoundedRectangle(x
,y
,w
,h
,radius
);
589 void wxGraphicsContext::StrokeLines( size_t n
, const wxPoint2DDouble
*points
)
592 wxGraphicsPath path
= CreatePath();
593 path
.MoveToPoint(points
[0].m_x
, points
[0].m_y
);
594 for ( size_t i
= 1; i
< n
; ++i
)
595 path
.AddLineToPoint( points
[i
].m_x
, points
[i
].m_y
);
599 void wxGraphicsContext::DrawLines( size_t n
, const wxPoint2DDouble
*points
, int fillStyle
)
602 wxGraphicsPath path
= CreatePath();
603 path
.MoveToPoint(points
[0].m_x
, points
[0].m_y
);
604 for ( size_t i
= 1; i
< n
; ++i
)
605 path
.AddLineToPoint( points
[i
].m_x
, points
[i
].m_y
);
606 DrawPath( path
, fillStyle
);
609 void wxGraphicsContext::StrokeLines( size_t n
, const wxPoint2DDouble
*beginPoints
, const wxPoint2DDouble
*endPoints
)
612 wxGraphicsPath path
= CreatePath();
613 for ( size_t i
= 0; i
< n
; ++i
)
615 path
.MoveToPoint(beginPoints
[i
].m_x
, beginPoints
[i
].m_y
);
616 path
.AddLineToPoint( endPoints
[i
].m_x
, endPoints
[i
].m_y
);
621 // create a 'native' matrix corresponding to these values
622 wxGraphicsMatrix
wxGraphicsContext::CreateMatrix( wxDouble a
, wxDouble b
, wxDouble c
, wxDouble d
,
623 wxDouble tx
, wxDouble ty
) const
625 return GetRenderer()->CreateMatrix(a
,b
,c
,d
,tx
,ty
);
628 wxGraphicsPath
wxGraphicsContext::CreatePath() const
630 return GetRenderer()->CreatePath();
633 wxGraphicsPen
wxGraphicsContext::CreatePen(const wxPen
& pen
) const
635 return GetRenderer()->CreatePen(pen
);
638 wxGraphicsBrush
wxGraphicsContext::CreateBrush(const wxBrush
& brush
) const
640 return GetRenderer()->CreateBrush(brush
);
643 // sets the brush to a linear gradient, starting at (x1,y1) with color c1 to (x2,y2) with color c2
644 wxGraphicsBrush
wxGraphicsContext::CreateLinearGradientBrush( wxDouble x1
, wxDouble y1
, wxDouble x2
, wxDouble y2
,
645 const wxColour
&c1
, const wxColour
&c2
) const
647 return GetRenderer()->CreateLinearGradientBrush(x1
,y1
,x2
,y2
,c1
,c2
);
650 // sets the brush to a radial gradient originating at (xo,yc) with color oColor and ends on a circle around (xc,yc)
651 // with radius r and color cColor
652 wxGraphicsBrush
wxGraphicsContext::CreateRadialGradientBrush( wxDouble xo
, wxDouble yo
, wxDouble xc
, wxDouble yc
, wxDouble radius
,
653 const wxColour
&oColor
, const wxColour
&cColor
) const
655 return GetRenderer()->CreateRadialGradientBrush(xo
,yo
,xc
,yc
,radius
,oColor
,cColor
);
659 wxGraphicsFont
wxGraphicsContext::CreateFont( const wxFont
&font
, const wxColour
&col
) const
661 return GetRenderer()->CreateFont(font
,col
);
664 wxGraphicsContext
* wxGraphicsContext::Create( const wxWindowDC
& dc
)
666 return wxGraphicsRenderer::GetDefaultRenderer()->CreateContext(dc
);
669 wxGraphicsContext
* wxGraphicsContext::CreateFromNative( void * context
)
671 return wxGraphicsRenderer::GetDefaultRenderer()->CreateContextFromNativeContext(context
);
674 wxGraphicsContext
* wxGraphicsContext::CreateFromNativeWindow( void * window
)
676 return wxGraphicsRenderer::GetDefaultRenderer()->CreateContextFromNativeWindow(window
);
679 wxGraphicsContext
* wxGraphicsContext::Create( wxWindow
* window
)
681 return wxGraphicsRenderer::GetDefaultRenderer()->CreateContext(window
);
684 //-----------------------------------------------------------------------------
685 // wxGraphicsRenderer
686 //-----------------------------------------------------------------------------
688 IMPLEMENT_ABSTRACT_CLASS(wxGraphicsRenderer
, wxObject
)
690 #endif // wxUSE_GRAPHICS_CONTEXT