1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/dcbase.cpp 
   3 // Purpose:     generic methods of the wxDC Class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) wxWidgets team 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  28 #include "wx/dcclient.h" 
  29 #include "wx/dcmemory.h" 
  30 #include "wx/dcscreen.h" 
  31 #include "wx/dcprint.h" 
  32 #include "wx/prntbase.h" 
  36     #include "wx/module.h" 
  40     #include "wx/msw/dcclient.h" 
  41     #include "wx/msw/dcmemory.h" 
  42     #include "wx/msw/dcscreen.h" 
  46     #include "wx/gtk/dcclient.h" 
  47     #include "wx/gtk/dcmemory.h" 
  48     #include "wx/gtk/dcscreen.h" 
  49 #elif defined(__WXGTK__) 
  50     #include "wx/gtk1/dcclient.h" 
  51     #include "wx/gtk1/dcmemory.h" 
  52     #include "wx/gtk1/dcscreen.h" 
  56     #include "wx/osx/dcclient.h" 
  57     #include "wx/osx/dcmemory.h" 
  58     #include "wx/osx/dcscreen.h" 
  62     #include "wx/os2/dcclient.h" 
  63     #include "wx/os2/dcmemory.h" 
  64     #include "wx/os2/dcscreen.h" 
  68     #include "wx/cocoa/dcclient.h" 
  69     #include "wx/cocoa/dcmemory.h" 
  70     #include "wx/cocoa/dcscreen.h" 
  74     #include "wx/motif/dcclient.h" 
  75     #include "wx/motif/dcmemory.h" 
  76     #include "wx/motif/dcscreen.h" 
  80     #include "wx/x11/dcclient.h" 
  81     #include "wx/x11/dcmemory.h" 
  82     #include "wx/x11/dcscreen.h" 
  86     #include "wx/dfb/dcclient.h" 
  87     #include "wx/dfb/dcmemory.h" 
  88     #include "wx/dfb/dcscreen.h" 
  92     #include "wx/palmos/dcclient.h" 
  93     #include "wx/palmos/dcmemory.h" 
  94     #include "wx/palmos/dcscreen.h" 
  97 //---------------------------------------------------------------------------- 
  99 //---------------------------------------------------------------------------- 
 101 wxDCFactory 
*wxDCFactory::m_factory 
= NULL
; 
 103 void wxDCFactory::Set(wxDCFactory 
*factory
) 
 110 wxDCFactory 
*wxDCFactory::Get() 
 113         m_factory 
= new wxNativeDCFactory
; 
 118 class wxDCFactoryCleanupModule 
: public wxModule
 
 121     virtual bool OnInit() { return true; } 
 122     virtual void OnExit() { wxDCFactory::Set(NULL
); } 
 125     DECLARE_DYNAMIC_CLASS(wxDCFactoryCleanupModule
) 
 128 IMPLEMENT_DYNAMIC_CLASS(wxDCFactoryCleanupModule
, wxModule
) 
 130 //----------------------------------------------------------------------------- 
 132 //----------------------------------------------------------------------------- 
 134 wxDCImpl
* wxNativeDCFactory::CreateWindowDC( wxWindowDC 
*owner
, wxWindow 
*window 
) 
 136     wxDCImpl 
* const impl 
= new wxWindowDCImpl( owner
, window 
); 
 137     impl
->InheritAttributes(window
); 
 141 wxDCImpl
* wxNativeDCFactory::CreateClientDC( wxClientDC 
*owner
, wxWindow 
*window 
) 
 143     wxDCImpl 
* const impl 
= new wxClientDCImpl( owner
, window 
); 
 144     impl
->InheritAttributes(window
); 
 148 wxDCImpl
* wxNativeDCFactory::CreatePaintDC( wxPaintDC 
*owner
, wxWindow 
*window 
) 
 150     wxDCImpl 
* const impl 
= new wxPaintDCImpl( owner
, window 
); 
 151     impl
->InheritAttributes(window
); 
 155 wxDCImpl
* wxNativeDCFactory::CreateMemoryDC( wxMemoryDC 
*owner 
) 
 157     return new wxMemoryDCImpl( owner 
); 
 160 wxDCImpl
* wxNativeDCFactory::CreateMemoryDC(wxMemoryDC 
*owner
, wxBitmap
& bitmap
) 
 162     // the bitmap may be modified when it's selected into a memory DC so make 
 163     // sure changing this bitmap doesn't affect any other shallow copies of it 
 164     // (see wxMemoryDC::SelectObject()) 
 166     // notice that we don't provide any ctor equivalent to SelectObjectAsSource 
 167     // method because this should be rarely needed and easy to work around by 
 168     // using the default ctor and calling SelectObjectAsSource itself 
 172     return new wxMemoryDCImpl(owner
, bitmap
); 
 175 wxDCImpl
* wxNativeDCFactory::CreateMemoryDC( wxMemoryDC 
*owner
, wxDC 
*dc 
) 
 177     return new wxMemoryDCImpl( owner
, dc 
); 
 180 wxDCImpl
* wxNativeDCFactory::CreateScreenDC( wxScreenDC 
*owner 
) 
 182     return new wxScreenDCImpl( owner 
); 
 185 #if wxUSE_PRINTING_ARCHITECTURE 
 186 wxDCImpl 
*wxNativeDCFactory::CreatePrinterDC( wxPrinterDC 
*owner
, const wxPrintData 
&data 
) 
 188     wxPrintFactory 
*factory 
= wxPrintFactory::GetFactory(); 
 189     return factory
->CreatePrinterDCImpl( owner
, data 
); 
 193 //----------------------------------------------------------------------------- 
 195 //----------------------------------------------------------------------------- 
 197 IMPLEMENT_ABSTRACT_CLASS(wxWindowDC
, wxDC
) 
 199 wxWindowDC::wxWindowDC(wxWindow 
*win
) 
 200           : wxDC(wxDCFactory::Get()->CreateWindowDC(this, win
)) 
 204 //----------------------------------------------------------------------------- 
 206 //----------------------------------------------------------------------------- 
 208 IMPLEMENT_ABSTRACT_CLASS(wxClientDC
, wxWindowDC
) 
 210 wxClientDC::wxClientDC(wxWindow 
*win
) 
 211           : wxWindowDC(wxDCFactory::Get()->CreateClientDC(this, win
)) 
 215 //----------------------------------------------------------------------------- 
 217 //----------------------------------------------------------------------------- 
 219 IMPLEMENT_DYNAMIC_CLASS(wxMemoryDC
, wxDC
) 
 221 wxMemoryDC::wxMemoryDC() 
 222           : wxDC(wxDCFactory::Get()->CreateMemoryDC(this)) 
 226 wxMemoryDC::wxMemoryDC(wxBitmap
& bitmap
) 
 227           : wxDC(wxDCFactory::Get()->CreateMemoryDC(this, bitmap
)) 
 231 wxMemoryDC::wxMemoryDC(wxDC 
*dc
) 
 232           : wxDC(wxDCFactory::Get()->CreateMemoryDC(this, dc
)) 
 236 void wxMemoryDC::SelectObject(wxBitmap
& bmp
) 
 238     // make sure that the given wxBitmap is not sharing its data with other 
 239     // wxBitmap instances as its contents will be modified by any drawing 
 240     // operation done on this DC 
 244     GetImpl()->DoSelect(bmp
); 
 247 void wxMemoryDC::SelectObjectAsSource(const wxBitmap
& bmp
) 
 249     GetImpl()->DoSelect(bmp
); 
 252 const wxBitmap
& wxMemoryDC::GetSelectedBitmap() const 
 254     return GetImpl()->GetSelectedBitmap(); 
 257 wxBitmap
& wxMemoryDC::GetSelectedBitmap() 
 259     return GetImpl()->GetSelectedBitmap(); 
 263 //----------------------------------------------------------------------------- 
 265 //----------------------------------------------------------------------------- 
 267 IMPLEMENT_ABSTRACT_CLASS(wxPaintDC
, wxClientDC
) 
 269 wxPaintDC::wxPaintDC(wxWindow 
*win
) 
 270          : wxClientDC(wxDCFactory::Get()->CreatePaintDC(this, win
)) 
 274 //----------------------------------------------------------------------------- 
 276 //----------------------------------------------------------------------------- 
 278 IMPLEMENT_DYNAMIC_CLASS(wxScreenDC
, wxWindowDC
) 
 280 wxScreenDC::wxScreenDC() 
 281           : wxDC(wxDCFactory::Get()->CreateScreenDC(this)) 
 285 //----------------------------------------------------------------------------- 
 287 //----------------------------------------------------------------------------- 
 289 #if wxUSE_PRINTING_ARCHITECTURE 
 291 IMPLEMENT_DYNAMIC_CLASS(wxPrinterDC
, wxDC
) 
 293 wxPrinterDC::wxPrinterDC() 
 294            : wxDC(wxDCFactory::Get()->CreatePrinterDC(this, wxPrintData())) 
 298 wxPrinterDC::wxPrinterDC(const wxPrintData
& data
) 
 299            : wxDC(wxDCFactory::Get()->CreatePrinterDC(this, data
)) 
 303 wxRect 
wxPrinterDC::GetPaperRect() const 
 305     return GetImpl()->GetPaperRect(); 
 308 int wxPrinterDC::GetResolution() const 
 310     return GetImpl()->GetResolution(); 
 313 #endif // wxUSE_PRINTING_ARCHITECTURE 
 315 //----------------------------------------------------------------------------- 
 317 //----------------------------------------------------------------------------- 
 319 IMPLEMENT_ABSTRACT_CLASS(wxDCImpl
, wxObject
) 
 321 wxDCImpl::wxDCImpl( wxDC 
*owner 
) 
 323         , m_colour(wxColourDisplay()) 
 327         , m_isBBoxValid(false) 
 328         , m_logicalOriginX(0), m_logicalOriginY(0) 
 329         , m_deviceOriginX(0), m_deviceOriginY(0) 
 330         , m_deviceLocalOriginX(0), m_deviceLocalOriginY(0) 
 331         , m_logicalScaleX(1.0), m_logicalScaleY(1.0) 
 332         , m_userScaleX(1.0), m_userScaleY(1.0) 
 333         , m_scaleX(1.0), m_scaleY(1.0) 
 334         , m_signX(1), m_signY(1) 
 335         , m_minX(0), m_minY(0), m_maxX(0), m_maxY(0) 
 336         , m_clipX1(0), m_clipY1(0), m_clipX2(0), m_clipY2(0) 
 337         , m_logicalFunction(wxCOPY
) 
 338         , m_backgroundMode(wxBRUSHSTYLE_TRANSPARENT
) 
 339         , m_mappingMode(wxMM_TEXT
) 
 342         , m_backgroundBrush() 
 343         , m_textForegroundColour(*wxBLACK
) 
 344         , m_textBackgroundColour(*wxWHITE
) 
 348         , m_hasCustomPalette(false) 
 349 #endif // wxUSE_PALETTE 
 353     m_mm_to_pix_x 
= (double)wxGetDisplaySize().GetWidth() / 
 354                     (double)wxGetDisplaySizeMM().GetWidth(); 
 355     m_mm_to_pix_y 
= (double)wxGetDisplaySize().GetHeight() / 
 356                     (double)wxGetDisplaySizeMM().GetHeight(); 
 362 wxDCImpl::~wxDCImpl() 
 366 // ---------------------------------------------------------------------------- 
 367 // coordinate conversions and transforms 
 368 // ---------------------------------------------------------------------------- 
 370 wxCoord 
wxDCImpl::DeviceToLogicalX(wxCoord x
) const 
 372     return wxRound( (double)((x 
- m_deviceOriginX 
- m_deviceLocalOriginX
) * m_signX
) / m_scaleX 
) + m_logicalOriginX 
; 
 375 wxCoord 
wxDCImpl::DeviceToLogicalY(wxCoord y
) const 
 377     return wxRound( (double)((y 
- m_deviceOriginY 
- m_deviceLocalOriginY
) * m_signY
) / m_scaleY 
) + m_logicalOriginY 
; 
 380 wxCoord 
wxDCImpl::DeviceToLogicalXRel(wxCoord x
) const 
 382     return wxRound((double)(x
) / m_scaleX
); 
 385 wxCoord 
wxDCImpl::DeviceToLogicalYRel(wxCoord y
) const 
 387     return wxRound((double)(y
) / m_scaleY
); 
 390 wxCoord 
wxDCImpl::LogicalToDeviceX(wxCoord x
) const 
 392     return wxRound( (double)((x 
- m_logicalOriginX
) * m_signX
) * m_scaleX
) + m_deviceOriginX 
+ m_deviceLocalOriginX
; 
 395 wxCoord 
wxDCImpl::LogicalToDeviceY(wxCoord y
) const 
 397     return wxRound( (double)((y 
- m_logicalOriginY
) * m_signY
) * m_scaleY
) + m_deviceOriginY 
+ m_deviceLocalOriginY
; 
 400 wxCoord 
wxDCImpl::LogicalToDeviceXRel(wxCoord x
) const 
 402     return wxRound((double)(x
) * m_scaleX
); 
 405 wxCoord 
wxDCImpl::LogicalToDeviceYRel(wxCoord y
) const 
 407     return wxRound((double)(y
) * m_scaleY
); 
 410 void wxDCImpl::ComputeScaleAndOrigin() 
 412     m_scaleX 
= m_logicalScaleX 
* m_userScaleX
; 
 413     m_scaleY 
= m_logicalScaleY 
* m_userScaleY
; 
 416 void wxDCImpl::SetMapMode( int mode 
) 
 421           SetLogicalScale( twips2mm
*m_mm_to_pix_x
, twips2mm
*m_mm_to_pix_y 
); 
 424           SetLogicalScale( pt2mm
*m_mm_to_pix_x
, pt2mm
*m_mm_to_pix_y 
); 
 427           SetLogicalScale( m_mm_to_pix_x
, m_mm_to_pix_y 
); 
 430           SetLogicalScale( m_mm_to_pix_x
/10.0, m_mm_to_pix_y
/10.0 ); 
 434           SetLogicalScale( 1.0, 1.0 ); 
 437     m_mappingMode 
= mode
; 
 440 void wxDCImpl::SetUserScale( double x
, double y 
) 
 442     // allow negative ? -> no 
 445     ComputeScaleAndOrigin(); 
 448 void wxDCImpl::SetLogicalScale( double x
, double y 
) 
 453     ComputeScaleAndOrigin(); 
 456 void wxDCImpl::SetLogicalOrigin( wxCoord x
, wxCoord y 
) 
 458     m_logicalOriginX 
= x 
* m_signX
; 
 459     m_logicalOriginY 
= y 
* m_signY
; 
 460     ComputeScaleAndOrigin(); 
 463 void wxDCImpl::SetDeviceOrigin( wxCoord x
, wxCoord y 
) 
 467     ComputeScaleAndOrigin(); 
 470 void wxDCImpl::SetDeviceLocalOrigin( wxCoord x
, wxCoord y 
) 
 472     m_deviceLocalOriginX 
= x
; 
 473     m_deviceLocalOriginY 
= y
; 
 474     ComputeScaleAndOrigin(); 
 477 void wxDCImpl::SetAxisOrientation( bool xLeftRight
, bool yBottomUp 
) 
 479     // only wxPostScripDC has m_signX = -1, we override SetAxisOrientation there 
 480     // wxWidgets 2.9: no longer override it 
 481     m_signX 
= (xLeftRight 
?  1 : -1); 
 482     m_signY 
= (yBottomUp  
? -1 :  1); 
 483     ComputeScaleAndOrigin(); 
 487 // Each element of the widths array will be the width of the string up to and 
 488 // including the corresponding character in text.  This is the generic 
 489 // implementation, the port-specific classes should do this with native APIs 
 490 // if available and if faster.  Note: pango_layout_index_to_pos is much slower 
 491 // than calling GetTextExtent!! 
 498     FontWidthCache() : m_scaleX(1), m_widths(NULL
) { } 
 499     ~FontWidthCache() { delete []m_widths
; } 
 504             m_widths 
= new int[FWC_SIZE
]; 
 506         memset(m_widths
, 0, sizeof(int)*FWC_SIZE
); 
 514 static FontWidthCache s_fontWidthCache
; 
 516 bool wxDCImpl::DoGetPartialTextExtents(const wxString
& text
, wxArrayInt
& widths
) const 
 520     const size_t len 
= text
.length(); 
 524     // reset the cache if font or horizontal scale have changed 
 525     if ( !s_fontWidthCache
.m_widths 
|| 
 526          !wxIsSameDouble(s_fontWidthCache
.m_scaleX
, m_scaleX
) || 
 527          (s_fontWidthCache
.m_font 
!= GetFont()) ) 
 529         s_fontWidthCache
.Reset(); 
 530         s_fontWidthCache
.m_font 
= GetFont(); 
 531         s_fontWidthCache
.m_scaleX 
= m_scaleX
; 
 534     // Calculate the position of each character based on the widths of 
 535     // the previous characters 
 537     for ( size_t i 
= 0; i 
< len
; i
++ ) 
 539         const wxChar c 
= text
[i
]; 
 540         unsigned int c_int 
= (unsigned int)c
; 
 542         if ((c_int 
< FWC_SIZE
) && (s_fontWidthCache
.m_widths
[c_int
] != 0)) 
 544             w 
= s_fontWidthCache
.m_widths
[c_int
]; 
 548             DoGetTextExtent(c
, &w
, &h
); 
 549             if (c_int 
< FWC_SIZE
) 
 550                 s_fontWidthCache
.m_widths
[c_int
] = w
; 
 554         widths
[i
] = totalWidth
; 
 560 void wxDCImpl::GetMultiLineTextExtent(const wxString
& text
, 
 564                                       const wxFont 
*font
) const 
 566     wxCoord widthTextMax 
= 0, widthLine
, 
 567             heightTextTotal 
= 0, heightLineDefault 
= 0, heightLine 
= 0; 
 570     for ( wxString::const_iterator pc 
= text
.begin(); ; ++pc 
) 
 572         if ( pc 
== text
.end() || *pc 
== _T('\n') ) 
 574             if ( curLine
.empty() ) 
 576                 // we can't use GetTextExtent - it will return 0 for both width 
 577                 // and height and an empty line should count in height 
 580                 // assume that this line has the same height as the previous 
 582                 if ( !heightLineDefault 
) 
 583                     heightLineDefault 
= heightLine
; 
 585                 if ( !heightLineDefault 
) 
 587                     // but we don't know it yet - choose something reasonable 
 588                     DoGetTextExtent(_T("W"), NULL
, &heightLineDefault
, 
 592                 heightTextTotal 
+= heightLineDefault
; 
 596                 DoGetTextExtent(curLine
, &widthLine
, &heightLine
, 
 598                 if ( widthLine 
> widthTextMax 
) 
 599                     widthTextMax 
= widthLine
; 
 600                 heightTextTotal 
+= heightLine
; 
 603             if ( pc 
== text
.end() ) 
 621         *y 
= heightTextTotal
; 
 626 void wxDCImpl::DoDrawCheckMark(wxCoord x1
, wxCoord y1
, 
 627                                wxCoord width
, wxCoord height
) 
 629     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 631     wxCoord x2 
= x1 
+ width
, 
 634     // the pen width is calibrated to give 3 for width == height == 10 
 635     wxDCPenChanger 
pen( *m_owner
, wxPen(GetTextForeground(), (width 
+ height 
+ 1)/7)); 
 637     // we're drawing a scaled version of wx/generic/tick.xpm here 
 638     wxCoord x3 
= x1 
+ (4*width
) / 10,   // x of the tick bottom 
 639             y3 
= y1 
+ height 
/ 2;       // y of the left tick branch 
 640     DoDrawLine(x1
, y3
, x3
, y2
); 
 641     DoDrawLine(x3
, y2
, x2
, y1
); 
 643     CalcBoundingBox(x1
, y1
); 
 644     CalcBoundingBox(x2
, y2
); 
 648 wxDCImpl::DoStretchBlit(wxCoord xdest
, wxCoord ydest
, 
 649                         wxCoord dstWidth
, wxCoord dstHeight
, 
 651                         wxCoord xsrc
, wxCoord ysrc
, 
 652                         wxCoord srcWidth
, wxCoord srcHeight
, 
 658     wxCHECK_MSG( srcWidth 
&& srcHeight 
&& dstWidth 
&& dstHeight
, false, 
 659                  _T("invalid blit size") ); 
 661     // emulate the stretching by modifying the DC scale 
 662     double xscale 
= (double)srcWidth
/dstWidth
, 
 663            yscale 
= (double)srcHeight
/dstHeight
; 
 665     double xscaleOld
, yscaleOld
; 
 666     GetUserScale(&xscaleOld
, &yscaleOld
); 
 667     SetUserScale(xscaleOld
/xscale
, yscaleOld
/yscale
); 
 669     bool rc 
= DoBlit(wxCoord(xdest
*xscale
), wxCoord(ydest
*yscale
), 
 670                      wxCoord(dstWidth
*xscale
), wxCoord(dstHeight
*yscale
), 
 672                      xsrc
, ysrc
, rop
, useMask
, xsrcMask
, ysrcMask
); 
 674     SetUserScale(xscaleOld
, yscaleOld
); 
 679 void wxDCImpl::DrawLines(const wxPointList 
*list
, wxCoord xoffset
, wxCoord yoffset
) 
 681     int n 
= list
->GetCount(); 
 682     wxPoint 
*points 
= new wxPoint
[n
]; 
 685     for ( wxPointList::compatibility_iterator node 
= list
->GetFirst(); node
; node 
= node
->GetNext(), i
++ ) 
 687         wxPoint 
*point 
= node
->GetData(); 
 688         points
[i
].x 
= point
->x
; 
 689         points
[i
].y 
= point
->y
; 
 692     DoDrawLines(n
, points
, xoffset
, yoffset
); 
 697 void wxDCImpl::DrawPolygon(const wxPointList 
*list
, 
 698                            wxCoord xoffset
, wxCoord yoffset
, 
 701     int n 
= list
->GetCount(); 
 702     wxPoint 
*points 
= new wxPoint
[n
]; 
 705     for ( wxPointList::compatibility_iterator node 
= list
->GetFirst(); node
; node 
= node
->GetNext(), i
++ ) 
 707         wxPoint 
*point 
= node
->GetData(); 
 708         points
[i
].x 
= point
->x
; 
 709         points
[i
].y 
= point
->y
; 
 712     DoDrawPolygon(n
, points
, xoffset
, yoffset
, fillStyle
); 
 718 wxDCImpl::DoDrawPolyPolygon(int n
, 
 721                             wxCoord xoffset
, wxCoord yoffset
, 
 726         DoDrawPolygon(count
[0], points
, xoffset
, yoffset
, fillStyle
); 
 734     for (i 
= j 
= lastOfs 
= 0; i 
< n
; i
++) 
 739     pts 
= new wxPoint
[j
+n
-1]; 
 740     for (i 
= 0; i 
< j
; i
++) 
 742     for (i 
= 2; i 
<= n
; i
++) 
 744         lastOfs 
-= count
[n
-i
]; 
 745         pts
[j
++] = pts
[lastOfs
]; 
 749     SetPen(wxPen(*wxBLACK
, 0, wxPENSTYLE_TRANSPARENT
)); 
 750     DoDrawPolygon(j
, pts
, xoffset
, yoffset
, fillStyle
); 
 752     for (i 
= j 
= 0; i 
< n
; i
++) 
 754         DoDrawLines(count
[i
], pts
+j
, xoffset
, yoffset
); 
 762 void wxDCImpl::DrawSpline(wxCoord x1
, wxCoord y1
, 
 763                           wxCoord x2
, wxCoord y2
, 
 764                           wxCoord x3
, wxCoord y3
) 
 766     wxPoint points
[] = { wxPoint(x1
, y1
), wxPoint(x2
, y2
), wxPoint(x3
, y3
) }; 
 767     DrawSpline(WXSIZEOF(points
), points
); 
 770 void wxDCImpl::DrawSpline(int n
, wxPoint points
[]) 
 773     for ( int i 
= 0; i 
< n
; i
++ ) 
 774         list
.Append(&points
[i
]); 
 779 // ----------------------------------- spline code ---------------------------------------- 
 781 void wx_quadratic_spline(double a1
, double b1
, double a2
, double b2
, 
 782                          double a3
, double b3
, double a4
, double b4
); 
 783 void wx_clear_stack(); 
 784 int wx_spline_pop(double *x1
, double *y1
, double *x2
, double *y2
, double *x3
, 
 785         double *y3
, double *x4
, double *y4
); 
 786 void wx_spline_push(double x1
, double y1
, double x2
, double y2
, double x3
, double y3
, 
 787           double x4
, double y4
); 
 788 static bool wx_spline_add_point(double x
, double y
); 
 789 static void wx_spline_draw_point_array(wxDC 
*dc
); 
 791 wxPointList wx_spline_point_list
; 
 793 #define                half(z1, z2)        ((z1+z2)/2.0) 
 796 /* iterative version */ 
 798 void wx_quadratic_spline(double a1
, double b1
, double a2
, double b2
, double a3
, double b3
, double a4
, 
 801     register double  xmid
, ymid
; 
 802     double           x1
, y1
, x2
, y2
, x3
, y3
, x4
, y4
; 
 805     wx_spline_push(a1
, b1
, a2
, b2
, a3
, b3
, a4
, b4
); 
 807     while (wx_spline_pop(&x1
, &y1
, &x2
, &y2
, &x3
, &y3
, &x4
, &y4
)) { 
 808         xmid 
= (double)half(x2
, x3
); 
 809         ymid 
= (double)half(y2
, y3
); 
 810         if (fabs(x1 
- xmid
) < THRESHOLD 
&& fabs(y1 
- ymid
) < THRESHOLD 
&& 
 811             fabs(xmid 
- x4
) < THRESHOLD 
&& fabs(ymid 
- y4
) < THRESHOLD
) { 
 812             wx_spline_add_point( x1
, y1 
); 
 813             wx_spline_add_point( xmid
, ymid 
); 
 815             wx_spline_push(xmid
, ymid
, (double)half(xmid
, x3
), (double)half(ymid
, y3
), 
 816                  (double)half(x3
, x4
), (double)half(y3
, y4
), x4
, y4
); 
 817             wx_spline_push(x1
, y1
, (double)half(x1
, x2
), (double)half(y1
, y2
), 
 818                  (double)half(x2
, xmid
), (double)half(y2
, ymid
), xmid
, ymid
); 
 823 /* utilities used by spline drawing routines */ 
 825 typedef struct wx_spline_stack_struct 
{ 
 826     double           x1
, y1
, x2
, y2
, x3
, y3
, x4
, y4
; 
 829 #define         SPLINE_STACK_DEPTH             20 
 830 static Stack    wx_spline_stack
[SPLINE_STACK_DEPTH
]; 
 831 static Stack   
*wx_stack_top
; 
 832 static int      wx_stack_count
; 
 834 void wx_clear_stack() 
 836     wx_stack_top 
= wx_spline_stack
; 
 840 void wx_spline_push(double x1
, double y1
, double x2
, double y2
, double x3
, double y3
, double x4
, double y4
) 
 842     wx_stack_top
->x1 
= x1
; 
 843     wx_stack_top
->y1 
= y1
; 
 844     wx_stack_top
->x2 
= x2
; 
 845     wx_stack_top
->y2 
= y2
; 
 846     wx_stack_top
->x3 
= x3
; 
 847     wx_stack_top
->y3 
= y3
; 
 848     wx_stack_top
->x4 
= x4
; 
 849     wx_stack_top
->y4 
= y4
; 
 854 int wx_spline_pop(double *x1
, double *y1
, double *x2
, double *y2
, 
 855                   double *x3
, double *y3
, double *x4
, double *y4
) 
 857     if (wx_stack_count 
== 0) 
 861     *x1 
= wx_stack_top
->x1
; 
 862     *y1 
= wx_stack_top
->y1
; 
 863     *x2 
= wx_stack_top
->x2
; 
 864     *y2 
= wx_stack_top
->y2
; 
 865     *x3 
= wx_stack_top
->x3
; 
 866     *y3 
= wx_stack_top
->y3
; 
 867     *x4 
= wx_stack_top
->x4
; 
 868     *y4 
= wx_stack_top
->y4
; 
 872 static bool wx_spline_add_point(double x
, double y
) 
 874     wxPoint 
*point 
= new wxPoint( wxRound(x
), wxRound(y
) ); 
 875     wx_spline_point_list
.Append(point 
); 
 879 static void wx_spline_draw_point_array(wxDC 
*dc
) 
 881     dc
->DrawLines(&wx_spline_point_list
, 0, 0 ); 
 882     wxPointList::compatibility_iterator node 
= wx_spline_point_list
.GetFirst(); 
 885         wxPoint 
*point 
= node
->GetData(); 
 887         wx_spline_point_list
.Erase(node
); 
 888         node 
= wx_spline_point_list
.GetFirst(); 
 892 void wxDCImpl::DoDrawSpline( const wxPointList 
*points 
) 
 894     wxCHECK_RET( IsOk(), wxT("invalid window dc") ); 
 897     double           cx1
, cy1
, cx2
, cy2
, cx3
, cy3
, cx4
, cy4
; 
 898     double           x1
, y1
, x2
, y2
; 
 900     wxPointList::compatibility_iterator node 
= points
->GetFirst(); 
 905     p 
= (wxPoint 
*)node
->GetData(); 
 910     node 
= node
->GetNext(); 
 915     cx1 
= (double)((x1 
+ x2
) / 2); 
 916     cy1 
= (double)((y1 
+ y2
) / 2); 
 917     cx2 
= (double)((cx1 
+ x2
) / 2); 
 918     cy2 
= (double)((cy1 
+ y2
) / 2); 
 920     wx_spline_add_point(x1
, y1
); 
 922     while ((node 
= node
->GetNext()) 
 933         cx4 
= (double)(x1 
+ x2
) / 2; 
 934         cy4 
= (double)(y1 
+ y2
) / 2; 
 935         cx3 
= (double)(x1 
+ cx4
) / 2; 
 936         cy3 
= (double)(y1 
+ cy4
) / 2; 
 938         wx_quadratic_spline(cx1
, cy1
, cx2
, cy2
, cx3
, cy3
, cx4
, cy4
); 
 942         cx2 
= (double)(cx1 
+ x2
) / 2; 
 943         cy2 
= (double)(cy1 
+ y2
) / 2; 
 946     wx_spline_add_point( cx1
, cy1 
); 
 947     wx_spline_add_point( x2
, y2 
); 
 949     wx_spline_draw_point_array( m_owner 
); 
 952 #endif // wxUSE_SPLINES 
 956 void wxDCImpl::DoGradientFillLinear(const wxRect
& rect
, 
 957                                     const wxColour
& initialColour
, 
 958                                     const wxColour
& destColour
, 
 959                                     wxDirection nDirection
) 
 962     wxPen oldPen 
= m_pen
; 
 963     wxBrush oldBrush 
= m_brush
; 
 965     wxUint8 nR1 
= initialColour
.Red(); 
 966     wxUint8 nG1 
= initialColour
.Green(); 
 967     wxUint8 nB1 
= initialColour
.Blue(); 
 968     wxUint8 nR2 
= destColour
.Red(); 
 969     wxUint8 nG2 
= destColour
.Green(); 
 970     wxUint8 nB2 
= destColour
.Blue(); 
 973     if ( nDirection 
== wxEAST 
|| nDirection 
== wxWEST 
) 
 975         wxInt32 x 
= rect
.GetWidth(); 
 976         wxInt32 w 
= x
;              // width of area to shade 
 977         wxInt32 xDelta 
= w
/256;     // height of one shade bend 
 985                 nR 
= nR1 
- (nR1
-nR2
)*(w
-x
)/w
; 
 987                 nR 
= nR1 
+ (nR2
-nR1
)*(w
-x
)/w
; 
 990                 nG 
= nG1 
- (nG1
-nG2
)*(w
-x
)/w
; 
 992                 nG 
= nG1 
+ (nG2
-nG1
)*(w
-x
)/w
; 
 995                 nB 
= nB1 
- (nB1
-nB2
)*(w
-x
)/w
; 
 997                 nB 
= nB1 
+ (nB2
-nB1
)*(w
-x
)/w
; 
 999             wxColour 
colour(nR
,nG
,nB
); 
1000             SetPen(wxPen(colour
, 1, wxPENSTYLE_SOLID
)); 
1001             SetBrush(wxBrush(colour
)); 
1002             if(nDirection 
== wxEAST
) 
1003                 DoDrawRectangle(rect
.GetRight()-x
-xDelta
+1, rect
.GetTop(), 
1004                         xDelta
, rect
.GetHeight()); 
1005             else //nDirection == wxWEST 
1006                 DoDrawRectangle(rect
.GetLeft()+x
, rect
.GetTop(), 
1007                         xDelta
, rect
.GetHeight()); 
1010     else  // nDirection == wxNORTH || nDirection == wxSOUTH 
1012         wxInt32 y 
= rect
.GetHeight(); 
1013         wxInt32 w 
= y
;              // height of area to shade 
1014         wxInt32 yDelta 
= w
/255;     // height of one shade bend 
1022                 nR 
= nR1 
- (nR1
-nR2
)*(w
-y
)/w
; 
1024                 nR 
= nR1 
+ (nR2
-nR1
)*(w
-y
)/w
; 
1027                 nG 
= nG1 
- (nG1
-nG2
)*(w
-y
)/w
; 
1029                 nG 
= nG1 
+ (nG2
-nG1
)*(w
-y
)/w
; 
1032                 nB 
= nB1 
- (nB1
-nB2
)*(w
-y
)/w
; 
1034                 nB 
= nB1 
+ (nB2
-nB1
)*(w
-y
)/w
; 
1036             wxColour 
colour(nR
,nG
,nB
); 
1037             SetPen(wxPen(colour
, 1, wxPENSTYLE_SOLID
)); 
1038             SetBrush(wxBrush(colour
)); 
1039             if(nDirection 
== wxNORTH
) 
1040                 DoDrawRectangle(rect
.GetLeft(), rect
.GetTop()+y
, 
1041                         rect
.GetWidth(), yDelta
); 
1042             else //nDirection == wxSOUTH 
1043                 DoDrawRectangle(rect
.GetLeft(), rect
.GetBottom()-y
-yDelta
+1, 
1044                         rect
.GetWidth(), yDelta
); 
1052 void wxDCImpl::DoGradientFillConcentric(const wxRect
& rect
, 
1053                                       const wxColour
& initialColour
, 
1054                                       const wxColour
& destColour
, 
1055                                       const wxPoint
& circleCenter
) 
1057     //save the old pen color 
1058     wxColour oldPenColour 
= m_pen
.GetColour(); 
1060     wxUint8 nR1 
= destColour
.Red(); 
1061     wxUint8 nG1 
= destColour
.Green(); 
1062     wxUint8 nB1 
= destColour
.Blue(); 
1063     wxUint8 nR2 
= initialColour
.Red(); 
1064     wxUint8 nG2 
= initialColour
.Green(); 
1065     wxUint8 nB2 
= initialColour
.Blue(); 
1070     wxInt32 cx 
= rect
.GetWidth() / 2; 
1071     wxInt32 cy 
= rect
.GetHeight() / 2; 
1079     wxInt32 nCircleOffX 
= circleCenter
.x 
- (rect
.GetWidth() / 2); 
1080     wxInt32 nCircleOffY 
= circleCenter
.y 
- (rect
.GetHeight() / 2); 
1082     for ( wxInt32 x 
= 0; x 
< rect
.GetWidth(); x
++ ) 
1084         for ( wxInt32 y 
= 0; y 
< rect
.GetHeight(); y
++ ) 
1086             //get color difference 
1087             wxInt32 nGradient 
= ((nRadius 
- 
1089                                     pow((double)(x 
- cx 
- nCircleOffX
), 2) + 
1090                                     pow((double)(y 
- cy 
- nCircleOffY
), 2) 
1091                                   )) * 100) / nRadius
; 
1093             //normalize Gradient 
1098             nR 
= (wxUint8
)(nR1 
+ ((nR2 
- nR1
) * nGradient 
/ 100)); 
1099             nG 
= (wxUint8
)(nG1 
+ ((nG2 
- nG1
) * nGradient 
/ 100)); 
1100             nB 
= (wxUint8
)(nB1 
+ ((nB2 
- nB1
) * nGradient 
/ 100)); 
1103             m_pen
.SetColour(wxColour(nR
,nG
,nB
)); 
1104             DoDrawPoint(x 
+ rect
.GetLeft(), y 
+ rect
.GetTop()); 
1107     //return old pen color 
1108     m_pen
.SetColour(oldPenColour
); 
1111 void wxDCImpl::InheritAttributes(wxWindow 
*win
) 
1113     wxCHECK_RET( win
, "window can't be NULL" ); 
1115     SetFont(win
->GetFont()); 
1116     SetTextForeground(win
->GetForegroundColour()); 
1117     SetTextBackground(win
->GetBackgroundColour()); 
1118     SetBackground(wxBrush(win
->GetBackgroundColour())); 
1121 //----------------------------------------------------------------------------- 
1123 //----------------------------------------------------------------------------- 
1125 IMPLEMENT_ABSTRACT_CLASS(wxDC
, wxObject
) 
1127 void wxDC::DrawLabel(const wxString
& text
, 
1128                          const wxBitmap
& bitmap
, 
1132                          wxRect 
*rectBounding
) 
1134     // find the text position 
1135     wxCoord widthText
, heightText
, heightLine
; 
1136     GetMultiLineTextExtent(text
, &widthText
, &heightText
, &heightLine
); 
1138     wxCoord width
, height
; 
1141         width 
= widthText 
+ bitmap
.GetWidth(); 
1142         height 
= bitmap
.GetHeight(); 
1147         height 
= heightText
; 
1151     if ( alignment 
& wxALIGN_RIGHT 
) 
1153         x 
= rect
.GetRight() - width
; 
1155     else if ( alignment 
& wxALIGN_CENTRE_HORIZONTAL 
) 
1157         x 
= (rect
.GetLeft() + rect
.GetRight() + 1 - width
) / 2; 
1159     else // alignment & wxALIGN_LEFT 
1164     if ( alignment 
& wxALIGN_BOTTOM 
) 
1166         y 
= rect
.GetBottom() - height
; 
1168     else if ( alignment 
& wxALIGN_CENTRE_VERTICAL 
) 
1170         y 
= (rect
.GetTop() + rect
.GetBottom() + 1 - height
) / 2; 
1172     else // alignment & wxALIGN_TOP 
1177     // draw the bitmap first 
1183         DrawBitmap(bitmap
, x
, y
, true /* use mask */); 
1185         wxCoord offset 
= bitmap
.GetWidth() + 4; 
1189         y 
+= (height 
- heightText
) / 2; 
1192     // we will draw the underscore under the accel char later 
1193     wxCoord startUnderscore 
= 0, 
1197     // split the string into lines and draw each of them separately 
1199     for ( wxString::const_iterator pc 
= text
.begin(); ; ++pc 
) 
1201         if ( pc 
== text
.end() || *pc 
== '\n' ) 
1203             int xRealStart 
= x
; // init it here to avoid compielr warnings 
1205             if ( !curLine
.empty() ) 
1207                 // NB: can't test for !(alignment & wxALIGN_LEFT) because 
1208                 //     wxALIGN_LEFT is 0 
1209                 if ( alignment 
& (wxALIGN_RIGHT 
| wxALIGN_CENTRE_HORIZONTAL
) ) 
1212                     GetTextExtent(curLine
, &widthLine
, NULL
); 
1214                     if ( alignment 
& wxALIGN_RIGHT 
) 
1216                         xRealStart 
+= width 
- widthLine
; 
1218                     else // if ( alignment & wxALIGN_CENTRE_HORIZONTAL ) 
1220                         xRealStart 
+= (width 
- widthLine
) / 2; 
1223                 //else: left aligned, nothing to do 
1225                 DrawText(curLine
, xRealStart
, y
); 
1230             // do we have underscore in this line? we can check yUnderscore 
1231             // because it is set below to just y + heightLine if we do 
1232             if ( y 
== yUnderscore 
) 
1234                 // adjust the horz positions to account for the shift 
1235                 startUnderscore 
+= xRealStart
; 
1236                 endUnderscore 
+= xRealStart
; 
1239             if ( pc 
== text
.end() ) 
1244         else // not end of line 
1246             if ( pc 
- text
.begin() == indexAccel 
) 
1248                 // remeber to draw underscore here 
1249                 GetTextExtent(curLine
, &startUnderscore
, NULL
); 
1251                 GetTextExtent(curLine
, &endUnderscore
, NULL
); 
1253                 yUnderscore 
= y 
+ heightLine
; 
1262     // draw the underscore if found 
1263     if ( startUnderscore 
!= endUnderscore 
) 
1265         // it should be of the same colour as text 
1266         SetPen(wxPen(GetTextForeground(), 0, wxPENSTYLE_SOLID
)); 
1270         DrawLine(startUnderscore
, yUnderscore
, endUnderscore
, yUnderscore
); 
1273     // return bounding rect if requested 
1276         *rectBounding 
= wxRect(x
, y 
- heightText
, widthText
, heightText
); 
1279     CalcBoundingBox(x0
, y0
); 
1280     CalcBoundingBox(x0 
+ width0
, y0 
+ height
); 
1283 #if WXWIN_COMPATIBILITY_2_8 
1284     // for compatibility with the old code when wxCoord was long everywhere 
1285 void wxDC::GetTextExtent(const wxString
& string
, 
1288                        long *externalLeading
, 
1289                        const wxFont 
*theFont
) const 
1291         wxCoord x2
, y2
, descent2
, externalLeading2
; 
1292         m_pimpl
->DoGetTextExtent(string
, &x2
, &y2
, 
1293                         &descent2
, &externalLeading2
, 
1300             *descent 
= descent2
; 
1301         if ( externalLeading 
) 
1302             *externalLeading 
= externalLeading2
; 
1305 void wxDC::GetLogicalOrigin(long *x
, long *y
) const 
1308         m_pimpl
->DoGetLogicalOrigin(&x2
, &y2
); 
1315 void wxDC::GetDeviceOrigin(long *x
, long *y
) const 
1318         m_pimpl
->DoGetDeviceOrigin(&x2
, &y2
); 
1325 void wxDC::GetClippingBox(long *x
, long *y
, long *w
, long *h
) const 
1327         wxCoord xx
,yy
,ww
,hh
; 
1328         m_pimpl
->DoGetClippingBox(&xx
, &yy
, &ww
, &hh
); 
1335 #endif  // WXWIN_COMPATIBILITY_2_8 
1338 Notes for wxWidgets DrawEllipticArcRot(...) 
1340 wxDCBase::DrawEllipticArcRot(...) draws a rotated elliptic arc or an ellipse. 
1341 It uses wxDCBase::CalculateEllipticPoints(...) and wxDCBase::Rotate(...), 
1344 All methods are generic, so they can be implemented in wxDCBase. 
1345 DoDrawEllipticArcRot(...) is virtual, so it can be called from deeper 
1346 methods like (WinCE) wxDC::DoDrawArc(...). 
1348 CalculateEllipticPoints(...) fills a given list of wxPoints with some points 
1349 of an elliptic arc. The algorithm is pixel-based: In every row (in flat 
1350 parts) or every column (in steep parts) only one pixel is calculated. 
1351 Trigonometric calculation (sin, cos, tan, atan) is only done if the 
1352 starting angle is not equal to the ending angle. The calculation of the 
1353 pixels is done using simple arithmetic only and should perform not too 
1354 bad even on devices without floating point processor. I didn't test this yet. 
1356 Rotate(...) rotates a list of point pixel-based, you will see rounding errors. 
1357 For instance: an ellipse rotated 180 degrees is drawn 
1358 slightly different from the original. 
1360 The points are then moved to an array and used to draw a polyline and/or polygon 
1361 (with center added, the pie). 
1362 The result looks quite similar to the native ellipse, only e few pixels differ. 
1364 The performance on a desktop system (Athlon 1800, WinXP) is about 7 times 
1365 slower as DrawEllipse(...), which calls the native API. 
1366 An rotated ellipse outside the clipping region takes nearly the same time, 
1367 while an native ellipse outside takes nearly no time to draw. 
1369 If you draw an arc with this new method, you will see the starting and ending angles 
1370 are calculated properly. 
1371 If you use DrawEllipticArc(...), you will see they are only correct for circles 
1372 and not properly calculated for ellipses. 
1375 p.lenhard@t-online.de 
1379 void wxDCImpl::DoDrawEllipticArcRot( wxCoord x
, wxCoord y
, 
1380                                      wxCoord w
, wxCoord h
, 
1381                                      double sa
, double ea
, double angle 
) 
1385     CalculateEllipticPoints( &list
, x
, y
, w
, h
, sa
, ea 
); 
1386     Rotate( &list
, angle
, wxPoint( x
+w
/2, y
+h
/2 ) ); 
1388     // Add center (for polygon/pie) 
1389     list
.Append( new wxPoint( x
+w
/2, y
+h
/2 ) ); 
1391     // copy list into array and delete list elements 
1392     int n 
= list
.GetCount(); 
1393     wxPoint 
*points 
= new wxPoint
[n
]; 
1395     wxPointList::compatibility_iterator node
; 
1396     for ( node 
= list
.GetFirst(); node
; node 
= node
->GetNext(), i
++ ) 
1398         wxPoint 
*point 
= node
->GetData(); 
1399         points
[i
].x 
= point
->x
; 
1400         points
[i
].y 
= point
->y
; 
1404     // first draw the pie without pen, if necessary 
1405     if( GetBrush() != *wxTRANSPARENT_BRUSH 
) 
1407         wxPen 
tempPen( GetPen() ); 
1408         SetPen( *wxTRANSPARENT_PEN 
); 
1409         DoDrawPolygon( n
, points
, 0, 0 ); 
1413     // then draw the arc without brush, if necessary 
1414     if( GetPen() != *wxTRANSPARENT_PEN 
) 
1417         DoDrawLines( n
-1, points
, 0, 0 ); 
1422 } // DrawEllipticArcRot 
1424 void wxDCImpl::Rotate( wxPointList
* points
, double angle
, wxPoint center 
) 
1429         double dSinA 
= -sin(angle
*2.0*pi
/360.0); 
1430         double dCosA 
= cos(angle
*2.0*pi
/360.0); 
1431         wxPointList::compatibility_iterator node
; 
1432         for ( node 
= points
->GetFirst(); node
; node 
= node
->GetNext() ) 
1434             wxPoint
* point 
= node
->GetData(); 
1436             // transform coordinates, if necessary 
1437             if( center
.x 
) point
->x 
-= center
.x
; 
1438             if( center
.y 
) point
->y 
-= center
.y
; 
1440             // calculate rotation, rounding simply by implicit cast to integer 
1441             int xTemp 
= point
->x 
* dCosA 
- point
->y 
* dSinA
; 
1442             point
->y 
= point
->x 
* dSinA 
+ point
->y 
* dCosA
; 
1445             // back transform coordinates, if necessary 
1446             if( center
.x 
) point
->x 
+= center
.x
; 
1447             if( center
.y 
) point
->y 
+= center
.y
; 
1452 void wxDCImpl::CalculateEllipticPoints( wxPointList
* points
, 
1453                                         wxCoord xStart
, wxCoord yStart
, 
1454                                         wxCoord w
, wxCoord h
, 
1455                                         double sa
, double ea 
) 
1466     bool bUseAngles 
= false; 
1472     // decrement 1 pixel if ellipse is smaller than 2*a, 2*b 
1474     if( 2*a 
== w 
) decrX 
= 1; 
1476     if( 2*b 
== h 
) decrY 
= 1; 
1478     wxCoord xCenter 
= xStart 
+ a
; 
1479     wxCoord yCenter 
= yStart 
+ b
; 
1480     // calculate data for start and end, if necessary 
1484         // normalisation of angles 
1485         while( sa
<0 ) sa 
+= 360; 
1486         while( ea
<0 ) ea 
+= 360; 
1487         while( sa
>=360 ) sa 
-= 360; 
1488         while( ea
>=360 ) ea 
-= 360; 
1489         // calculate quadrant numbers 
1490         if( sa 
> 270 ) sq 
= 3; 
1491         else if( sa 
> 180 ) sq 
= 2; 
1492         else if( sa 
> 90 ) sq 
= 1; 
1493         if( ea 
> 270 ) eq 
= 3; 
1494         else if( ea 
> 180 ) eq 
= 2; 
1495         else if( ea 
> 90 ) eq 
= 1; 
1496         sar 
= sa 
* pi 
/ 180.0; 
1497         ear 
= ea 
* pi 
/ 180.0; 
1498         // correct angle circle -> ellipse 
1499         sar 
= atan( -a
/(double)b 
* tan( sar 
) ); 
1500         if ( sq 
== 1 || sq 
== 2 ) sar 
+= pi
; 
1501         ear 
= atan( -a
/(double)b 
* tan( ear 
) ); 
1502         if ( eq 
== 1 || eq 
== 2 ) ear 
+= pi
; 
1503         // coordinates of points 
1504         xsa 
= xCenter 
+ a 
* cos( sar 
); 
1505         if( sq 
== 0 || sq 
== 3 ) xsa 
-= decrX
; 
1506         ysa 
= yCenter 
+ b 
* sin( sar 
); 
1507         if( sq 
== 2 || sq 
== 3 ) ysa 
-= decrY
; 
1508         xea 
= xCenter 
+ a 
* cos( ear 
); 
1509         if( eq 
== 0 || eq 
== 3 ) xea 
-= decrX
; 
1510         yea 
= yCenter 
+ b 
* sin( ear 
); 
1511         if( eq 
== 2 || eq 
== 3 ) yea 
-= decrY
; 
1513     // calculate c1 = b^2, c2 = b^2/a^2 with a = w/2, b = h/2 
1515     double c2 
= 2.0 / w
; 
1524     // Lists for quadrant 1 to 4 
1525     wxPointList pointsarray
[4]; 
1526     // Calculate points for first quadrant and set in all quadrants 
1527     for( x 
= 0; x 
<= a
; ++x 
) 
1532         bool bNewPoint 
= false; 
1533         while( y2 
> c1 
- c2 
* x2 
&& y 
> 0 ) 
1539         // old y now to big: set point with old y, old x 
1540         if( bNewPoint 
&& x
>1) 
1543             // remove points on the same line 
1544             pointsarray
[0].Insert( new wxPoint( xCenter 
+ x1 
- decrX
, yCenter 
- y_old 
) ); 
1545             pointsarray
[1].Append( new wxPoint( xCenter 
- x1
, yCenter 
- y_old 
) ); 
1546             pointsarray
[2].Insert( new wxPoint( xCenter 
- x1
, yCenter 
+ y_old 
- decrY 
) ); 
1547             pointsarray
[3].Append( new wxPoint( xCenter 
+ x1 
- decrX
, yCenter 
+ y_old 
- decrY 
) ); 
1549     } // calculate point 
1551     // Starting and/or ending points for the quadrants, first quadrant gets both. 
1552     pointsarray
[0].Insert( new wxPoint( xCenter 
+ a 
- decrX
, yCenter 
) ); 
1553     pointsarray
[0].Append( new wxPoint( xCenter
, yCenter 
- b 
) ); 
1554     pointsarray
[1].Append( new wxPoint( xCenter 
- a
, yCenter 
) ); 
1555     pointsarray
[2].Append( new wxPoint( xCenter
, yCenter 
+ b 
- decrY 
) ); 
1556     pointsarray
[3].Append( new wxPoint( xCenter 
+ a 
- decrX
, yCenter 
) ); 
1558     // copy quadrants in original list 
1561         // Copy the right part of the points in the lists 
1562         // and delete the wxPoints, because they do not leave this method. 
1563         points
->Append( new wxPoint( xsa
, ysa 
) ); 
1565         bool bStarted 
= false; 
1566         bool bReady 
= false; 
1567         bool bForceTurn 
= ( sq 
== eq 
&& sa 
> ea 
); 
1570             wxPointList::compatibility_iterator node
; 
1571             for( node 
= pointsarray
[q
].GetFirst(); node
; node 
= node
->GetNext() ) 
1573                 // once: go to starting point in start quadrant 
1576                       node
->GetData()->x 
< xsa
+1 && q 
<= 1 
1578                       node
->GetData()->x 
> xsa
-1 && q 
>= 2 
1585                 // copy point, if not at ending point 
1588                     if( q 
!= eq 
|| bForceTurn
 
1590                         ( (wxPoint
*) node
->GetData() )->x 
> xea
+1 && q 
<= 1 
1592                         ( (wxPoint
*) node
->GetData() )->x 
< xea
-1 && q 
>= 2 
1596                         wxPoint
* pPoint 
= new wxPoint( *(node
->GetData()) ); 
1597                         points
->Append( pPoint 
); 
1599                     else if( q 
== eq 
&& !bForceTurn 
|| node
->GetData()->x 
== xea
) 
1609         } // while not bReady 
1610         points
->Append( new wxPoint( xea
, yea 
) ); 
1613         for( q 
= 0; q 
< 4; ++q 
) 
1615             wxPointList::compatibility_iterator node
; 
1616             for( node 
= pointsarray
[q
].GetFirst(); node
; node 
= node
->GetNext() ) 
1618                 wxPoint 
*p 
= node
->GetData(); 
1625         wxPointList::compatibility_iterator node
; 
1626         // copy whole ellipse, wxPoints will be deleted outside 
1627         for( node 
= pointsarray
[0].GetFirst(); node
; node 
= node
->GetNext() ) 
1629             wxPoint 
*p 
= node
->GetData(); 
1630             points
->Append( p 
); 
1632         for( node 
= pointsarray
[1].GetFirst(); node
; node 
= node
->GetNext() ) 
1634             wxPoint 
*p 
= node
->GetData(); 
1635             points
->Append( p 
); 
1637         for( node 
= pointsarray
[2].GetFirst(); node
; node 
= node
->GetNext() ) 
1639             wxPoint 
*p 
= node
->GetData(); 
1640             points
->Append( p 
); 
1642         for( node 
= pointsarray
[3].GetFirst(); node
; node 
= node
->GetNext() ) 
1644             wxPoint 
*p 
= node
->GetData(); 
1645             points
->Append( p 
); 
1648 } // CalculateEllipticPoints 
1650 #endif // __WXWINCE__ 
1652 float wxDCImpl::GetFontPointSizeAdjustment(float dpi
) 
1654     // wxMSW has long-standing bug where wxFont point size is interpreted as 
1655     // "pixel size corresponding to given point size *on screen*". In other 
1656     // words, on a typical 600dpi printer and a typical 96dpi screen, fonts 
1657     // are ~6 times smaller when printing. Unfortunately, this bug is so severe 
1658     // that *all* printing code has to account for it and consequently, other 
1659     // ports need to emulate this bug too: 
1660     const wxSize screenPPI 
= wxGetDisplayPPI(); 
1661     return float(screenPPI
.y
) / dpi
;