1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/mgl/dc.cpp 
   4 // Author:      Vaclav Slavik 
   7 // Copyright:   (c) 2001-2002 SciTech Software, Inc. (www.scitechsoft.com) 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  11 // =========================================================================== 
  13 // =========================================================================== 
  15 // --------------------------------------------------------------------------- 
  17 // --------------------------------------------------------------------------- 
  19 // For compilers that support precompilation, includes "wx.h". 
  20 #include "wx/wxprec.h" 
  28     #include "wx/dcmemory.h" 
  32 #include "wx/fontutil.h" 
  33 #include "wx/encinfo.h" 
  34 #include "wx/fontmap.h" 
  35 #include "wx/mgl/private.h" 
  42 //----------------------------------------------------------------------------- 
  44 //----------------------------------------------------------------------------- 
  46 const double RAD2DEG        
= 180.0 / M_PI
; 
  49 //----------------------------------------------------------------------------- 
  51 //----------------------------------------------------------------------------- 
  53 const ushort STIPPLE_wxDOT        
= 0x5555/* - - - - - - - -*/; 
  54 const ushort STIPPLE_wxLONG_DASH  
= 0xF0F0/*    ----    ----*/; 
  55 const ushort STIPPLE_wxSHORT_DASH 
= 0xCCCC/*--  --  --  --  */; 
  56 const ushort STIPPLE_wxDOT_DASH   
= 0x3939/*  ---  -  ---  -*/; 
  57 const ushort STIPPLE_wxSOLID      
= 0xFFFF/*----------------*/; 
  59 #define PATTERN_ROW(b7,b6,b5,b4,b3,b2,b1,b0) \ 
  60             ((b7 << 7) | (b6 << 6) | (b5 << 5) | (b4 << 4) | \ 
  61              (b3 << 3) | (b2 << 2) | (b1 << 1) | b0) 
  63 static pattern_t PATTERN_wxFDIAGONAL_HATCH 
= {{ 
  64        PATTERN_ROW(1,0,0,0,0,0,0,0), 
  65        PATTERN_ROW(0,1,0,0,0,0,0,0), 
  66        PATTERN_ROW(0,0,1,0,0,0,0,0), 
  67        PATTERN_ROW(0,0,0,1,0,0,0,0), 
  68        PATTERN_ROW(0,0,0,0,1,0,0,0), 
  69        PATTERN_ROW(0,0,0,0,0,1,0,0), 
  70        PATTERN_ROW(0,0,0,0,0,0,1,0), 
  71        PATTERN_ROW(0,0,0,0,0,0,0,1), 
  74 static pattern_t PATTERN_wxCROSSDIAG_HATCH 
= {{ 
  75        PATTERN_ROW(1,0,0,0,0,0,0,1), 
  76        PATTERN_ROW(0,1,0,0,0,0,1,0), 
  77        PATTERN_ROW(0,0,1,0,0,1,0,0), 
  78        PATTERN_ROW(0,0,0,1,1,0,0,0), 
  79        PATTERN_ROW(0,0,0,1,1,0,0,0), 
  80        PATTERN_ROW(0,0,1,0,0,1,0,0), 
  81        PATTERN_ROW(0,1,0,0,0,0,1,0), 
  82        PATTERN_ROW(1,0,0,0,0,0,0,1), 
  85 static pattern_t PATTERN_wxBDIAGONAL_HATCH 
= {{ 
  86        PATTERN_ROW(0,0,0,0,0,0,0,1), 
  87        PATTERN_ROW(0,0,0,0,0,0,1,0), 
  88        PATTERN_ROW(0,0,0,0,0,1,0,0), 
  89        PATTERN_ROW(0,0,0,0,1,0,0,0), 
  90        PATTERN_ROW(0,0,0,1,0,0,0,0), 
  91        PATTERN_ROW(0,0,1,0,0,0,0,0), 
  92        PATTERN_ROW(0,1,0,0,0,0,0,0), 
  93        PATTERN_ROW(1,0,0,0,0,0,0,0), 
  96 static pattern_t PATTERN_wxCROSS_HATCH 
= {{ 
  97        PATTERN_ROW(0,0,0,1,0,0,0,0), 
  98        PATTERN_ROW(0,0,0,1,0,0,0,0), 
  99        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 100        PATTERN_ROW(1,1,1,1,1,1,1,1), 
 101        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 102        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 103        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 104        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 107 static pattern_t PATTERN_wxHORIZONTAL_HATCH 
= {{ 
 108        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 109        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 110        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 111        PATTERN_ROW(1,1,1,1,1,1,1,1), 
 112        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 113        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 114        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 115        PATTERN_ROW(0,0,0,0,0,0,0,0), 
 118 static pattern_t PATTERN_wxVERTICAL_HATCH 
= {{ 
 119        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 120        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 121        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 122        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 123        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 124        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 125        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 126        PATTERN_ROW(0,0,0,1,0,0,0,0), 
 131 //----------------------------------------------------------------------------- 
 133 //----------------------------------------------------------------------------- 
 135 IMPLEMENT_ABSTRACT_CLASS(wxDC
, wxDCBase
) 
 137 // Default constructor 
 143     m_ok 
= false; // must call SetMGLDevCtx() before using it 
 145     m_mm_to_pix_x 
= (double)wxGetDisplaySize().GetWidth() / 
 146                     (double)wxGetDisplaySizeMM().GetWidth(); 
 147     m_mm_to_pix_y 
= (double)wxGetDisplaySize().GetHeight() / 
 148                     (double)wxGetDisplaySizeMM().GetHeight(); 
 150     m_pen 
= *wxBLACK_PEN
; 
 151     m_font 
= *wxNORMAL_FONT
; 
 152     m_brush 
= *wxWHITE_BRUSH
; 
 153     m_penOfsX 
= m_penOfsY 
= 0; 
 155     m_penSelected 
= m_brushSelected 
= false; 
 156     m_downloadedPatterns
[0] = m_downloadedPatterns
[1] = false; 
 168 void wxDC::SetMGLDC(MGLDevCtx 
*mgldc
, bool OwnsMGLDC
) 
 170     if ( m_OwnsMGLDC 
&& m_MGLDC 
) 
 173     m_OwnsMGLDC 
= OwnsMGLDC
; 
 176     if ( !m_globalClippingRegion
.IsNull() ) 
 177         SetClippingRegion(m_globalClippingRegion
); 
 182 void wxDC::InitializeMGLDC() 
 184     if ( GetDepth() > 8 ) 
 186         wxCurrentDCSwitcher 
switcher(m_MGLDC
); // will go away with MGL6 
 187         m_MGLDC
->setFontBlendMode(MGL_AA_RGBBLEND
); 
 192 // --------------------------------------------------------------------------- 
 194 // --------------------------------------------------------------------------- 
 197 #define DO_SET_CLIPPING_BOX(rg)                      \ 
 199     wxRect rect = rg.GetBox();                       \ 
 200     m_clipX1 = (wxCoord) XDEV2LOG(rect.GetLeft());   \ 
 201     m_clipY1 = (wxCoord) YDEV2LOG(rect.GetTop());    \ 
 202     m_clipX2 = (wxCoord) XDEV2LOG(rect.GetRight());  \ 
 203     m_clipY2 = (wxCoord) YDEV2LOG(rect.GetBottom()); \ 
 206 void wxDC::DoSetClippingRegion(wxCoord cx
, wxCoord cy
, wxCoord cw
, wxCoord ch
) 
 208     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 210     wxRect 
rect(XLOG2DEV(cx
), YLOG2DEV(cy
), XLOG2DEVREL(cw
), YLOG2DEVREL(ch
)); 
 212     if ( !m_currentClippingRegion
.IsNull() ) 
 213         m_currentClippingRegion
.Intersect(rect
); 
 215         m_currentClippingRegion
.Union(rect
); 
 217     m_MGLDC
->setClipRegion(m_currentClippingRegion
.GetMGLRegion()); 
 220     DO_SET_CLIPPING_BOX(m_currentClippingRegion
) 
 223 void wxDC::DoSetDeviceClippingRegion(const wxRegion
& region
) 
 225     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 227     if ( region
.IsEmpty() ) 
 229         DestroyClippingRegion(); 
 235     // check if the DC is scaled or moved, and if yes, then 
 236     // convert rg to device coordinates: 
 237     if ( m_deviceOriginX 
!= 0 || m_deviceOriginY 
!= 0 || 
 238          m_logicalOriginX 
!= 0 || m_logicalOriginY 
!= 0 || 
 239          XLOG2DEVREL(500) != 500 || YLOG2DEVREL(500) != 500 ) 
 241         region_t 
*mrg 
= rg
.GetMGLRegion().rgnPointer(); 
 244         for (s 
= mrg
->spans
; s
; s 
= s
->next
) 
 246             s
->y 
= YLOG2DEV(s
->y
); 
 247             for (p 
= s
->seg
; p
; p 
= p
->next
) 
 248                 p
->x 
= XLOG2DEV(p
->x
); 
 252     if ( !m_currentClippingRegion
.IsNull() ) 
 253         m_currentClippingRegion
.Intersect(rg
); 
 255         m_currentClippingRegion
.Union(rg
); 
 257     m_MGLDC
->setClipRegion(m_currentClippingRegion
.GetMGLRegion()); 
 260     DO_SET_CLIPPING_BOX(m_currentClippingRegion
) 
 263 void wxDC::DestroyClippingRegion() 
 265     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 267     if ( !m_globalClippingRegion
.IsNull() ) 
 269         m_MGLDC
->setClipRegion(m_globalClippingRegion
.GetMGLRegion()); 
 270         m_currentClippingRegion 
= m_globalClippingRegion
; 
 275         m_MGLDC
->setClipRect(MGLRect(0, 0, m_MGLDC
->sizex()+1, m_MGLDC
->sizey()+1)); 
 277         m_currentClippingRegion
.Clear(); 
 281 // --------------------------------------------------------------------------- 
 282 // query capabilities 
 283 // --------------------------------------------------------------------------- 
 285 bool wxDC::CanDrawBitmap() const 
 290 bool wxDC::CanGetTextExtent() const 
 295 int wxDC::GetDepth() const 
 297     return m_MGLDC
->getBitsPerPixel(); 
 300 // --------------------------------------------------------------------------- 
 302 // --------------------------------------------------------------------------- 
 306     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 308     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 309     if ( m_backgroundBrush
.GetStyle() != wxTRANSPARENT 
) 
 312         wxBrush oldb 
= m_brush
; 
 313         SetBrush(m_backgroundBrush
); 
 316         m_MGLDC
->fillRect(0, 0, w
, h
); 
 321 extern bool wxDoFloodFill(wxDC 
*dc
, wxCoord x
, wxCoord y
, 
 322                           const wxColour 
& col
, wxFloodFillStyle style
); 
 324 bool wxDC::DoFloodFill(wxCoord x
, wxCoord y
, 
 325                        const wxColour
& col
, wxFloodFillStyle style
) 
 327     return wxDoFloodFill(this, x
, y
, col
, style
); 
 330 bool wxDC::DoGetPixel(wxCoord x
, wxCoord y
, wxColour 
*col
) const 
 332     wxCHECK_MSG( col
, false, _T("NULL colour parameter in wxDC::GetPixel")); 
 335     m_MGLDC
->unpackColorFast(m_MGLDC
->getPixel(XLOG2DEV(x
), YLOG2DEV(y
)), 
 341 void wxDC::DoCrossHair(wxCoord x
, wxCoord y
) 
 343     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 345     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 350         m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 351         if ( !m_penSelected 
) 
 353         wxCoord xx 
= XLOG2DEV(x
); 
 354         wxCoord yy 
= YLOG2DEV(y
); 
 355         m_MGLDC
->line(m_penOfsX
, yy 
+ m_penOfsY
, w
-1 + m_penOfsX
, yy 
+ m_penOfsY
); 
 356         m_MGLDC
->line(xx 
+ m_penOfsX
, m_penOfsY
, x 
+ m_penOfsX
, h
-1 + m_penOfsY
); 
 357         CalcBoundingBox(0, 0); 
 358         CalcBoundingBox(w
, h
); 
 362 void wxDC::DoDrawLine(wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
) 
 364     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 366     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 368         m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 369         if ( !m_penSelected 
) 
 371         m_MGLDC
->lineExt(XLOG2DEV(x1
) + m_penOfsX
, YLOG2DEV(y1
) + m_penOfsY
, 
 372                       XLOG2DEV(x2
) + m_penOfsX
, YLOG2DEV(y2
) + m_penOfsY
,FALSE
); 
 373         CalcBoundingBox(x1
, y1
); 
 374         CalcBoundingBox(x2
, y2
); 
 378 // Draws an arc of a circle, centred on (xc, yc), with starting point (x1, y1) 
 379 // and ending at (x2, y2) 
 380 void wxDC::DoDrawArc(wxCoord x1
, wxCoord y1
, 
 381                      wxCoord x2
, wxCoord y2
, 
 382                      wxCoord xc
, wxCoord yc
) 
 384     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 386     wxCoord xx1 
= XLOG2DEV(x1
); 
 387     wxCoord yy1 
= YLOG2DEV(y1
); 
 388     wxCoord xx2 
= XLOG2DEV(x2
); 
 389     wxCoord yy2 
= YLOG2DEV(y2
); 
 390     wxCoord xxc 
= XLOG2DEV(xc
); 
 391     wxCoord yyc 
= YLOG2DEV(yc
); 
 392     double dx 
= xx1 
- xxc
; 
 393     double dy 
= yy1 
- yyc
; 
 394     double radius 
= sqrt((double)(dx
*dx
+dy
*dy
)); 
 395     wxCoord r 
= (wxCoord
)radius
; 
 396     double radius1
, radius2
; 
 399     if (xx1 
== xx2 
&& yy1 
== yy2
) 
 404     else if (radius 
== 0.0) 
 406         radius1 
= radius2 
= 0.0; 
 410         radius1 
= (xx1 
- xxc 
== 0) ? 
 411             (yy1 
- yyc 
< 0) ? 90.0 : -90.0 : 
 412             -atan2(double(yy1
-yyc
), double(xx1
-xxc
)) * RAD2DEG
; 
 413         radius2 
= (xx2 
- xxc 
== 0) ? 
 414             (yy2 
- yyc 
< 0) ? 90.0 : -90.0 : 
 415             -atan2(double(yy2
-yyc
), double(xx2
-xxc
)) * RAD2DEG
; 
 417     wxCoord alpha1 
= wxCoord(radius1
); 
 418     wxCoord alpha2 
= alpha1 
+ wxCoord(radius2 
- radius1
); 
 419     while (alpha2 
<= 0) alpha2 
+= 360; 
 420     while (alpha1 
> 360) alpha1 
-= 360; 
 422     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 423     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 425         if ( !m_brushSelected 
) 
 427         m_MGLDC
->fillEllipseArc(xxc
, yyc
, r
, r
, alpha1
, alpha2
); 
 430     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 432         if ( !m_penSelected 
) 
 434         m_MGLDC
->ellipseArc(xxc 
+ m_penOfsX
, yyc 
+ m_penOfsY
, r
, r
, alpha1
, alpha2
); 
 437     CalcBoundingBox(xc 
- r
, yc 
- r
); 
 438     CalcBoundingBox(xc 
+ r
, yc 
+ r
); 
 441 void wxDC::DoDrawPoint(wxCoord x
, wxCoord y
) 
 443     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 445     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 447         m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 448         if ( !m_penSelected 
) 
 450         m_MGLDC
->pixel(XLOG2DEV(x
), YLOG2DEV(y
)); 
 451         CalcBoundingBox(x
, y
); 
 455 void wxDC::DoDrawPolygon(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
,wxPolygonFillMode 
WXUNUSED(fillStyle
)) 
 457     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 459     wxCoord xxoffset 
= XLOG2DEVREL(xoffset
), 
 460             yyoffset 
= YLOG2DEVREL(yoffset
); 
 461     MGLPoint 
*cpoints 
= new MGLPoint
[n
+1]; 
 462     for (int i 
= 0; i 
< n
; i
++) 
 464         CalcBoundingBox(points
[i
].x 
+ xoffset
, points
[i
].y 
+ yoffset
); 
 465         cpoints
[i
].x 
= (int)(XLOG2DEV(points
[i
].x
)); 
 466         cpoints
[i
].y 
= (int)(YLOG2DEV(points
[i
].y
)); 
 468     cpoints
[n
] = cpoints
[0]; 
 470     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 471     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 473         if ( !m_brushSelected 
) 
 475         m_MGLDC
->fillPolygon(n
, cpoints
, xxoffset
, yyoffset
); 
 478     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 480         if ( !m_penSelected 
) 
 482         if (m_penOfsX 
!= 0 || m_penOfsY 
!= 0) 
 484             for (int i 
= 0; i 
<= n
; i
++) 
 486                 cpoints
[i
].x 
+= m_penOfsX
; 
 487                 cpoints
[i
].y 
+= m_penOfsY
; 
 490         m_MGLDC
->polyLine(n
+1, cpoints
); 
 496 void wxDC::DoDrawLines(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
) 
 498     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 500     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 502         MGLPoint 
*cpoints 
= new MGLPoint
[n
]; 
 503         m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 504         if ( !m_penSelected 
) 
 506         for (int i 
= 0; i 
< n
; i
++) 
 508             CalcBoundingBox(points
[i
].x 
+ xoffset
, points
[i
].y 
+ yoffset
); 
 509             cpoints
[i
].x 
= (int)(XLOG2DEV(points
[i
].x 
+ xoffset
) /*+ m_penOfsX*/); 
 510             cpoints
[i
].y 
= (int)(YLOG2DEV(points
[i
].y 
+ yoffset
) /*+ m_penOfsY*/); 
 512         m_MGLDC
->polyLine(n
, cpoints
); 
 517 void wxDC::DoDrawRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
) 
 519     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 521     wxCoord xx 
= XLOG2DEV(x
); 
 522     wxCoord yy 
= YLOG2DEV(y
); 
 523     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 524     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 526     if ( ww 
== 0 || hh 
== 0 ) return; 
 539     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 540     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 542         if ( !m_brushSelected 
) 
 544         m_MGLDC
->fillRect(xx
, yy
, xx 
+ ww
, yy 
+ hh
); 
 547     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 549         if ( !m_penSelected 
) 
 552         m_MGLDC
->rect(xx 
+ m_penOfsX
, yy 
+ m_penOfsY
, 
 553                       xx 
+ ww 
+ m_penOfsX
, yy 
+ hh 
+ m_penOfsY
); 
 556     CalcBoundingBox(x
, y
); 
 557     CalcBoundingBox(x 
+ width
, y 
+ height
); 
 560 void wxDC::DoDrawRoundedRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double radius
) 
 562     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 565         radius 
= -radius 
* ((width 
< height
) ? width 
: height
); 
 567     wxCoord xx 
= XLOG2DEV(x
); 
 568     wxCoord yy 
= YLOG2DEV(y
); 
 569     wxCoord ww 
= m_signX 
* XLOG2DEVREL(width
); 
 570     wxCoord hh 
= m_signY 
* YLOG2DEVREL(height
); 
 571     wxCoord rr 
= XLOG2DEVREL((wxCoord
)radius
); 
 573     // CMB: handle -ve width and/or height 
 585     // CMB: if radius is zero use DrawRectangle() instead to avoid 
 586     // X drawing errors with small radii 
 589         DrawRectangle(x
, y
, width
, height
); 
 593     // CMB: draw nothing if transformed w or h is 0 
 594     if ( ww 
== 0 || hh 
== 0 ) return; 
 596     // CMB: ensure dd is not larger than rectangle otherwise we 
 597     // get an hour glass shape 
 599     if ( dd 
> ww 
) dd 
= ww
; 
 600     if ( dd 
> hh 
) dd 
= hh
; 
 603     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 604     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 606         if (!m_brushSelected
) 
 608         m_MGLDC
->fillRect(xx
+rr
, yy
, xx
+ww
-rr
, yy
+hh
); 
 609         m_MGLDC
->fillRect(xx
, yy
+rr
, xx
+ww
, yy
+hh
-rr
); 
 610         m_MGLDC
->fillEllipseArc(xx
+rr
, yy
+rr
, rr
, rr
, 90, 180); 
 611         m_MGLDC
->fillEllipseArc(xx
+ww
-rr
, yy
+rr
, rr
, rr
, 0, 90); 
 612         m_MGLDC
->fillEllipseArc(xx
+rr
, yy
+hh
-rr
, rr
, rr
, 180, 270); 
 613         m_MGLDC
->fillEllipseArc(xx
+ww
-rr
, yy
+hh
-rr
, rr
, rr
, 270, 0); 
 616     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 618         if ( !m_penSelected 
) 
 622         m_MGLDC
->line(xx
+rr
+1, yy
, xx
+ww
-rr
, yy
); 
 623         m_MGLDC
->ellipseArc(xx
+ww
-rr
, yy
+rr
, rr
, rr
, 0, 90); 
 624         m_MGLDC
->line(xx
+ww
, yy
+rr
+1, xx
+ww
, yy
+hh
-rr
); 
 625         m_MGLDC
->ellipseArc(xx
+ww
-rr
, yy
+hh
-rr
, rr
, rr
, 270, 0); 
 626         m_MGLDC
->line(xx
+ww
-rr
, yy
+hh
, xx
+rr
+1, yy
+hh
); 
 627         m_MGLDC
->ellipseArc(xx
+rr
, yy
+hh
-rr
, rr
, rr
, 180, 270); 
 628         m_MGLDC
->line(xx
, yy
+hh
-rr
, xx
, yy
+rr
+1); 
 629         m_MGLDC
->ellipseArc(xx
+rr
, yy
+rr
, rr
, rr
, 90, 180); 
 632     CalcBoundingBox(x
, y
); 
 633     CalcBoundingBox(x 
+ width
, y 
+ height
); 
 636 void wxDC::DoDrawEllipse(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
) 
 638     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 640     wxCoord x2 
= (x
+width
); 
 641     wxCoord y2 
= (y
+height
); 
 643     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 644     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 646         if ( !m_brushSelected 
) 
 648         MGLRect 
rect(XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
)); 
 649         m_MGLDC
->fillEllipse(rect
); 
 652     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 654         if ( !m_penSelected 
) 
 656         MGLRect 
rect(XLOG2DEV(x
) + m_penOfsX
, YLOG2DEV(y
) + m_penOfsY
, 
 657                      XLOG2DEV(x2
) + m_penOfsX
, YLOG2DEV(y2
) + m_penOfsY
); 
 658         m_MGLDC
->ellipse(rect
); 
 661     CalcBoundingBox(x
, y
); 
 662     CalcBoundingBox(x2
, y2
); 
 665 void wxDC::DoDrawEllipticArc(wxCoord x
,wxCoord y
,wxCoord w
,wxCoord h
,double sa
,double ea
) 
 667     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 672     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 673     if ( m_brush
.GetStyle() != wxTRANSPARENT 
) 
 675         if (!m_brushSelected
) SelectBrush(); 
 676         MGLRect 
rect(XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
)); 
 677         m_MGLDC
->fillEllipseArc(rect
, (int)sa
, (int)ea
); 
 680     if ( m_pen
.GetStyle() != wxTRANSPARENT 
) 
 682         if ( !m_penSelected 
) 
 684         MGLRect 
rect(XLOG2DEV(x
) + m_penOfsX
, YLOG2DEV(y
) + m_penOfsY
, 
 685                      XLOG2DEV(x2
) + m_penOfsX
, YLOG2DEV(y2
) + m_penOfsY
); 
 686         m_MGLDC
->ellipseArc(rect
, (int)sa
, (int)ea
); 
 689     CalcBoundingBox(x
, y
); 
 690     CalcBoundingBox(x2
, y2
); 
 693 void wxDC::DoDrawText(const wxString
& text
, wxCoord x
, wxCoord y
) 
 695     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 696     DrawAnyText(text
, x
, y
); 
 698     // update the bounding box 
 700     CalcBoundingBox(x
, y
); 
 701     GetTextExtent(text
, &w
, &h
); 
 702     CalcBoundingBox(x 
+ w
, y 
+ h
); 
 705 bool wxDC::SelectMGLFont() 
 707     if ( m_mglFont 
== NULL 
) 
 709         float scale 
= m_scaleY
; 
 710         bool antialiased 
= (GetDepth() > 8); 
 712         m_mglFont 
= m_font
.GetMGLfont_t(scale
, antialiased
); 
 713         wxCHECK_MSG( m_mglFont
, false, wxT("invalid font") ); 
 715         m_MGLDC
->useFont(m_mglFont
); 
 716         wxLogTrace("mgl_font", "useFont(%p)", m_mglFont
); 
 719         wxNativeEncodingInfo nativeEnc
; 
 720         wxFontEncoding encoding 
= m_font
.GetEncoding(); 
 721         if ( !wxGetNativeFontEncoding(encoding
, &nativeEnc
) || 
 722              !wxTestFontEncoding(nativeEnc
) ) 
 725             if ( !wxFontMapper::Get()->GetAltForEncoding(encoding
, &nativeEnc
) ) 
 728                 nativeEnc
.mglEncoding 
= MGL_ENCODING_ASCII
; 
 731         m_MGLDC
->setTextEncoding(nativeEnc
.mglEncoding
); 
 737 void wxDC::DrawAnyText(const wxString
& text
, wxCoord x
, wxCoord y
) 
 739     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
 744     wxCoord xx 
= XLOG2DEV(x
); 
 745     wxCoord yy 
= YLOG2DEV(y
); 
 747     m_MGLDC
->setLineStyle(MGL_LINE_STIPPLE
); 
 748     m_MGLDC
->setLineStipple(0xFFFF); 
 749     m_MGLDC
->setPenSize(1, 1); 
 750     m_MGLDC
->setPenStyle(MGL_BITMAP_SOLID
); 
 753     const wchar_t *c_text 
= text
.c_str(); 
 755     const char *c_text 
= text
.c_str(); 
 759     // FIXME_MGL - this is a temporary hack in absence of proper 
 760     //             implementation of solid text background in MGL. Once 
 761     //             the bug in MGL is fixed, this code should be nuked 
 762     //             immediately. Note that the code is not 100% correct; 
 763     //             it only works with wxCOPY logical function 
 764     if ( m_backgroundMode 
== wxSOLID 
) 
 766         int w 
= m_MGLDC
->textWidth(c_text
); 
 767         int h 
= m_MGLDC
->textHeight(); 
 768         m_MGLDC
->setColor(m_MGLDC
->packColorFast(m_textBackgroundColour
.Red(), 
 769                 m_textBackgroundColour
.Green(), m_textBackgroundColour
.Blue())); 
 770         m_MGLDC
->fillRect(xx
, yy
, xx
+w
, yy
+h
); 
 774     m_MGLDC
->setColor(m_MGLDC
->packColorFast(m_textForegroundColour
.Red(), 
 775             m_textForegroundColour
.Green(), m_textForegroundColour
.Blue())); 
 776     m_MGLDC
->setBackColor(m_MGLDC
->packColorFast(m_textBackgroundColour
.Red(), 
 777             m_textBackgroundColour
.Green(), m_textBackgroundColour
.Blue())); 
 779     m_MGLDC
->drawStr(xx
, yy
, c_text
); 
 782     if ( m_font
.GetUnderlined() ) 
 784         int x1 
= xx
, y1 
= yy
; 
 786         int w 
= m_MGLDC
->textWidth(c_text
); 
 787         m_MGLDC
->underScoreLocation(x1
, y1
, c_text
); 
 788         switch (m_MGLDC
->getTextDirection()) 
 790             case MGL_RIGHT_DIR
: x2 
= x1 
+ w
, y2 
= y1
; break; 
 791             case MGL_LEFT_DIR
:  x2 
= x1 
- w
, y2 
= y1
; break; 
 792             case MGL_UP_DIR
:    x2 
= x1
, y2 
= y1 
- w
; break; 
 793             case MGL_DOWN_DIR
:  x2 
= x1
, y2 
= y1 
+ w
; break; 
 795         m_MGLDC
->line(x1
, y1
, x2
, y2
); 
 798     m_penSelected 
= m_brushSelected 
= false; 
 801 void wxDC::DoDrawRotatedText(const wxString
& text
, 
 802                              wxCoord x
, wxCoord y
, 
 805     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
 809         DoDrawText(text
, x
, y
); 
 812     else if ( angle 
== 90.0 ) 
 813         m_MGLDC
->setTextDirection(MGL_UP_DIR
); 
 814     else if ( angle 
== 180.0 ) 
 815         m_MGLDC
->setTextDirection(MGL_LEFT_DIR
); 
 816     else if ( angle 
== 270.0 ) 
 817         m_MGLDC
->setTextDirection(MGL_DOWN_DIR
); 
 820         // FIXME_MGL -- implement once MGL supports it 
 821         wxFAIL_MSG(wxT("wxMGL only supports rotated text with angle 0,90,180 or 270")); 
 825     DrawAnyText(text
, x
, y
); 
 828     m_MGLDC
->setTextDirection(MGL_RIGHT_DIR
); 
 831 // --------------------------------------------------------------------------- 
 833 // --------------------------------------------------------------------------- 
 835 void wxDC::SelectMGLStipplePen(int style
) 
 841         case wxDOT
:        stipple 
= STIPPLE_wxDOT
;        break; 
 842         case wxLONG_DASH
:  stipple 
= STIPPLE_wxLONG_DASH
;  break; 
 843         case wxSHORT_DASH
: stipple 
= STIPPLE_wxSHORT_DASH
; break; 
 844         case wxDOT_DASH
:   stipple 
= STIPPLE_wxDOT_DASH
;   break; 
 845         default:           stipple 
= STIPPLE_wxSOLID
;      break; 
 848     m_MGLDC
->setLineStyle(MGL_LINE_STIPPLE
); 
 849     m_MGLDC
->setLineStipple(stipple
); 
 850     m_MGLDC
->setPenSize(1, 1); 
 851     m_MGLDC
->setPenStyle(MGL_BITMAP_SOLID
); 
 852     m_penOfsY 
= m_penOfsX 
= 0; 
 855 // Accepted valus of SelectMGLFatPen's 2nd argument 
 857     wxMGL_SELECT_FROM_PEN
, 
 858     wxMGL_SELECT_FROM_BRUSH
 
 861 void wxDC::SelectMGLFatPen(int style
, int flag
) 
 863     MGL_penStyleType penstyle
; 
 864     const pattern_t 
*pattern 
= NULL
; 
 865     pixpattern24_t 
*pixPattern 
= NULL
; 
 869     // Since MGL pens may be created from wxBrush or wxPen and we often 
 870     // switch between pens and brushes, we take advantage of MGL's ability 
 871     // to have multiple (pix)pattern_t's loaded. We always download pen 
 872     // to 0th slot and brush to 1st slot. 
 873     if ( flag 
== wxMGL_SELECT_FROM_PEN 
) 
 878     // compute pen's width: 
 879     if ( m_pen
.GetWidth() <= 1 ) 
 882         m_penOfsX 
= m_penOfsY 
= 0; 
 886         wx 
= (int)(0.5 + fabs((double) XLOG2DEVREL(m_pen
.GetWidth()))); 
 887         wy 
= (int)(0.5 + fabs((double) YLOG2DEVREL(m_pen
.GetWidth()))); 
 893     penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 896         case wxBDIAGONAL_HATCH
:  pattern 
= &PATTERN_wxBDIAGONAL_HATCH
; 
 897                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 899         case wxCROSSDIAG_HATCH
:  pattern 
= &PATTERN_wxCROSSDIAG_HATCH
; 
 900                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 902         case wxFDIAGONAL_HATCH
:  pattern 
= &PATTERN_wxFDIAGONAL_HATCH
; 
 903                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 905         case wxCROSS_HATCH
:      pattern 
= &PATTERN_wxCROSS_HATCH
; 
 906                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 908         case wxHORIZONTAL_HATCH
: pattern 
= &PATTERN_wxHORIZONTAL_HATCH
; 
 909                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 911         case wxVERTICAL_HATCH
:   pattern 
= &PATTERN_wxVERTICAL_HATCH
; 
 912                                  penstyle 
= MGL_BITMAP_TRANSPARENT
; 
 916             if ( flag 
== wxMGL_SELECT_FROM_PEN 
) 
 917                 pixPattern 
= (pixpattern24_t
*) m_pen
.GetPixPattern(); 
 919                 pixPattern 
= (pixpattern24_t
*) m_brush
.GetPixPattern(); 
 920             penstyle 
= MGL_PIXMAP
; 
 923         case wxSTIPPLE_MASK_OPAQUE
: 
 924             pattern 
= (pattern_t
*) m_brush
.GetMaskPattern(); 
 925             penstyle 
= MGL_BITMAP_OPAQUE
; 
 930             penstyle 
= MGL_BITMAP_SOLID
; break; 
 933     // ...and finally, pass the pen to MGL: 
 937         if ( !m_downloadedPatterns
[slot
] ) 
 939             m_MGLDC
->setPenBitmapPattern(slot
, pattern
); 
 940             m_downloadedPatterns
[slot
] = true; 
 942         m_MGLDC
->usePenBitmapPattern(slot
); 
 947         if ( !m_downloadedPatterns
[slot
] ) 
 955                     for (y 
= 0; y 
< 8; y
++) 
 956                         for (x 
= 0; x 
< 8; x
++) 
 957                             pix
.b8
.p
[x
][y
] = (uchar
)m_MGLDC
->packColorFast( 
 958                                                         pixPattern
->p
[x
][y
][0], 
 959                                                         pixPattern
->p
[x
][y
][1], 
 960                                                         pixPattern
->p
[x
][y
][2]); 
 964                     for (y 
= 0; y 
< 8; y
++) 
 965                         for (x 
= 0; x 
< 8; x
++) 
 966                             pix
.b16
.p
[x
][y
] = (M_uint16
)m_MGLDC
->packColorFast( 
 967                                                         pixPattern
->p
[x
][y
][0], 
 968                                                         pixPattern
->p
[x
][y
][1], 
 969                                                         pixPattern
->p
[x
][y
][2]); 
 972                     for (y 
= 0; y 
< 8; y
++) 
 973                         for (x 
= 0; x 
< 8; x
++) 
 974                             for (c 
= 0; c 
< 3; c
++) 
 975                                 pix
.b24
.p
[x
][y
][c
] = pixPattern
->p
[x
][y
][c
]; 
 978                     for (y 
= 0; y 
< 8; y
++) 
 979                         for (x 
= 0; x 
< 8; x
++) 
 980                             pix
.b32
.p
[x
][y
] = m_MGLDC
->packColorFast( 
 981                                                         pixPattern
->p
[x
][y
][0], 
 982                                                         pixPattern
->p
[x
][y
][1], 
 983                                                         pixPattern
->p
[x
][y
][2]); 
 986                     wxFAIL_MSG(_T("invalid DC depth")); 
 989             m_MGLDC
->setPenPixmapPattern(slot
, &pix
); 
 990             m_downloadedPatterns
[slot
] = true; 
 992         m_MGLDC
->usePenPixmapPattern(slot
); 
 995     m_MGLDC
->setLineStyle(MGL_LINE_PENSTYLE
); 
 996     m_MGLDC
->setPenStyle(penstyle
); 
 997     m_MGLDC
->setPenSize(wy
, wx
); 
1000 void wxDC::SelectPen() 
1002     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1004     wxColour
& clr 
= m_pen
.GetColour(); 
1005     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1006     m_MGLDC
->setColorRGB(clr
.Red(), clr
.Green(), clr
.Blue()); 
1008     switch (m_pen
.GetStyle()) 
1017             SelectMGLStipplePen(m_pen
.GetStyle()); 
1020         case wxBDIAGONAL_HATCH
: 
1021         case wxCROSSDIAG_HATCH
: 
1022         case wxFDIAGONAL_HATCH
: 
1024         case wxHORIZONTAL_HATCH
: 
1025         case wxVERTICAL_HATCH
: 
1026             SelectMGLFatPen(m_pen
.GetStyle(), wxMGL_SELECT_FROM_PEN
); 
1030             SelectMGLFatPen(m_pen
.GetStyle(), wxMGL_SELECT_FROM_PEN
); 
1036             if ( m_pen
.GetWidth() <= 1 ) 
1037                 SelectMGLStipplePen(wxSOLID
); 
1039                 SelectMGLFatPen(wxSOLID
, wxMGL_SELECT_FROM_PEN
); 
1042     m_penSelected 
= true; 
1043     m_brushSelected 
= false; 
1046 void wxDC::SelectBrush() 
1048     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1051     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1053     if ( m_brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE 
) 
1055         fg 
= m_textForegroundColour
; 
1056         bg 
= m_textBackgroundColour
; 
1060         fg 
= m_brush
.GetColour(); 
1061         bg 
= m_backgroundBrush
.GetColour(); 
1064     m_MGLDC
->setColorRGB(fg
.Red(), fg
.Green(), fg
.Blue()); 
1065     m_MGLDC
->setBackColor(m_MGLDC
->packColorFast(bg
.Red(), bg
.Green(), bg
.Blue())); 
1066     m_penSelected 
= false; 
1067     m_brushSelected 
= true; 
1069     SelectMGLFatPen(m_brush
.GetStyle(), wxMGL_SELECT_FROM_BRUSH
); 
1072 void wxDC::SetPen(const wxPen
& pen
) 
1074     if ( !pen
.Ok() ) return; 
1075     if ( m_pen 
== pen 
) return; 
1077     m_penSelected 
= false; 
1078     m_downloadedPatterns
[0] = false; 
1081 void wxDC::SetBrush(const wxBrush
& brush
) 
1083     if ( !brush
.Ok() ) return; 
1084     if ( m_brush 
== brush 
) return; 
1086     m_brushSelected 
= false; 
1087     m_downloadedPatterns
[1] = false; 
1090 void wxDC::SetPalette(const wxPalette
& palette
) 
1092     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1094     if ( !palette
.Ok() ) 
1096         if ( m_oldPalette
.Ok() ) 
1097             SetPalette(m_oldPalette
); 
1101     if ( palette
.IsSameAs(m_palette
) ) 
1104     m_oldPalette 
= m_palette
; 
1105     m_palette 
= palette
; 
1107     int cnt 
= m_palette
.GetColoursCount(); 
1108     palette_t 
*pal 
= m_palette
.GetMGLpalette_t(); 
1109     m_MGLDC
->setPalette(pal
, cnt
, 0); 
1110     m_MGLDC
->realizePalette(cnt
, 0, TRUE
); 
1113 void wxDC::SetFont(const wxFont
& font
) 
1122 void wxDC::SetBackground(const wxBrush
& brush
) 
1124     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1126     if (!brush
.Ok()) return; 
1128     m_backgroundBrush 
= brush
; 
1129     wxColour 
&clr 
= m_backgroundBrush
.GetColour(); 
1130     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1131     m_MGLDC
->setBackColor( 
1132              m_MGLDC
->packColorFast(clr
.Red(), clr
.Green(), clr
.Blue())); 
1135 void wxDC::SetBackgroundMode(int mode
) 
1137     m_backgroundMode 
= mode
; 
1138     if ( mode 
== wxSOLID 
) 
1139         m_MGLDC
->setBackMode(MGL_OPAQUE_BACKGROUND
); 
1141         m_MGLDC
->setBackMode(MGL_TRANSPARENT_BACKGROUND
); 
1144 void wxDC::SetLogicalFunction(wxRasterOperationMode function
) 
1146     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1148     m_logicalFunction 
= function
; 
1150     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1151     m_MGLDC
->setWriteMode(LogicalFunctionToMGLRop(m_logicalFunction
)); 
1154 int wxDC::LogicalFunctionToMGLRop(int logFunc
) const 
1156     MGL_writeModeType rop
; 
1160         case wxCLEAR
:        rop 
= MGL_R2_BLACK
;         break; 
1161         case wxXOR
:          rop 
= MGL_R2_XORSRC
;        break; 
1162         case wxINVERT
:       rop 
= MGL_R2_NOT
;           break; 
1163         case wxOR_REVERSE
:   rop 
= MGL_R2_MERGESRCNOT
;   break; 
1164         case wxAND_REVERSE
:  rop 
= MGL_R2_MASKSRCNOT
;    break; 
1165         case wxCOPY
:         rop 
= MGL_R2_COPYSRC
;       break; 
1166         case wxAND
:          rop 
= MGL_R2_MASKSRC
;       break; 
1167         case wxAND_INVERT
:   rop 
= MGL_R2_MASKNOTSRC
;    break; 
1168         case wxNO_OP
:        rop 
= MGL_R2_NOP
;           break; 
1169         case wxNOR
:          rop 
= MGL_R2_NOTMERGESRC
;   break; 
1170         case wxEQUIV
:        rop 
= MGL_R2_NOTXORSRC
;     break; 
1171         case wxSRC_INVERT
:   rop 
= MGL_R2_NOTCOPYSRC
;    break; 
1172         case wxOR_INVERT
:    rop 
= MGL_R2_MERGENOTSRC
;   break; 
1173         case wxNAND
:         rop 
= MGL_R2_NOTMASKSRC
;    break; 
1174         case wxOR
:           rop 
= MGL_R2_MERGESRC
;      break; 
1175         case wxSET
:          rop 
= MGL_R2_WHITE
;         break; 
1177            wxFAIL_MSG( wxT("unsupported logical function") ); 
1178            return MGL_REPLACE_MODE
; 
1183 bool wxDC::StartDoc(const wxString
& WXUNUSED(message
)) 
1185     // We might be previewing, so return true to let it continue. 
1193 void wxDC::StartPage() 
1197 void wxDC::EndPage() 
1201 // --------------------------------------------------------------------------- 
1203 // --------------------------------------------------------------------------- 
1205 wxCoord 
wxDC::GetCharHeight() const 
1207     wxCurrentDCSwitcher 
switcher(m_MGLDC
); 
1208     if ( !wxConstCast(this, wxDC
)->SelectMGLFont() ) return -1; 
1209     return YDEV2LOGREL(m_mglFont
->fontHeight
); 
1212 wxCoord 
wxDC::GetCharWidth() const 
1214     wxCurrentDCSwitcher 
switcher(m_MGLDC
); 
1215     if ( !wxConstCast(this, wxDC
)->SelectMGLFont() ) return -1; 
1216     // VS: wxT() is intentional, charWidth() has both char and wchar_t version 
1217     // VS: YDEV is corrent, it should *not* be XDEV, because font's are only 
1218     //     scaled according to m_scaleY 
1219     return YDEV2LOGREL(m_mglFont
->fontWidth
); 
1222 void wxDC::DoGetTextExtent(const wxString
& string
, wxCoord 
*x
, wxCoord 
*y
, 
1223                            wxCoord 
*descent
, wxCoord 
*externalLeading
, 
1224                            const wxFont 
*theFont
) const 
1228     if ( theFont 
!= NULL 
) 
1231         wxConstCast(this, wxDC
)->SetFont(*theFont
); 
1234     wxCurrentDCSwitcher 
switcher(m_MGLDC
); 
1235     if ( !wxConstCast(this, wxDC
)->SelectMGLFont() ) return; 
1238         // VS: YDEV is corrent, it should *not* be XDEV, because font's are 
1239         //     only scaled according to m_scaleY 
1240         *x 
= YDEV2LOGREL(m_MGLDC
->textWidth(string
.wc_str())); 
1242         *y 
= YDEV2LOGREL(m_MGLDC
->textHeight()); 
1244         *descent 
= YDEV2LOGREL(m_mglFont
->descent
); 
1245     if ( externalLeading 
) 
1246         *externalLeading 
= YDEV2LOGREL(m_mglFont
->leading
); 
1248     if ( theFont 
!= NULL 
) 
1249         wxConstCast(this, wxDC
)->SetFont(oldFont
); 
1253 // --------------------------------------------------------------------------- 
1255 // --------------------------------------------------------------------------- 
1257 void wxDC::ComputeScaleAndOrigin() 
1259     double newX 
= m_logicalScaleX 
* m_userScaleX
; 
1260     double newY 
= m_logicalScaleY 
* m_userScaleY
; 
1262     // make sure font will be reloaded before drawing: 
1263     if ( newY 
!= m_scaleY 
) 
1265     // make sure m_penOfs{X,Y} will be reevaluated before drawing: 
1266     if ( newY 
!= m_scaleY 
|| newX 
!= m_scaleX 
) 
1267         m_penSelected 
= false; 
1269     m_scaleX 
= newX
, m_scaleY 
= newY
; 
1272 void wxDC::DoGetSize(int *w
, int *h
) const 
1274     if (w
) *w 
= m_MGLDC
->sizex()+1; 
1275     if (h
) *h 
= m_MGLDC
->sizey()+1; 
1278 void wxDC::DoGetSizeMM(int *width
, int *height
) const 
1283     if ( width 
) *width 
= int(double(w
) / (m_userScaleX
*m_mm_to_pix_x
)); 
1284     if ( height 
) *height 
= int(double(h
) / (m_userScaleY
*m_mm_to_pix_y
)); 
1287 wxSize 
wxDC::GetPPI() const 
1289     return wxSize(int(double(m_mm_to_pix_x
) * inches2mm
), 
1290                   int(double(m_mm_to_pix_y
) * inches2mm
)); 
1294 // --------------------------------------------------------------------------- 
1296 // --------------------------------------------------------------------------- 
1298 bool wxDC::DoBlit(wxCoord xdest
, wxCoord ydest
, 
1299                   wxCoord width
, wxCoord height
, 
1300                   wxDC 
*source
, wxCoord xsrc
, wxCoord ysrc
, 
1301                   wxRasterOperationMode rop
, bool useMask
, 
1302                   wxCoord xsrcMask
, wxCoord ysrcMask
) 
1304     wxCHECK_MSG( Ok(), false, wxT("invalid dc") ); 
1305     wxCHECK_MSG( source
, false, wxT("invalid source dc") ); 
1307     // transform the source DC coords to the device ones 
1308     xsrc 
= source
->LogicalToDeviceX(xsrc
); 
1309     ysrc 
= source
->LogicalToDeviceY(ysrc
); 
1311     /* FIXME_MGL: use the mask origin when drawing transparently */ 
1312     if (xsrcMask 
== -1 && ysrcMask 
== -1) 
1314         xsrcMask 
= xsrc
; ysrcMask 
= ysrc
; 
1318         xsrcMask 
= source
->LogicalToDeviceX(xsrcMask
); 
1319         ysrcMask 
= source
->LogicalToDeviceY(ysrcMask
); 
1322     CalcBoundingBox(xdest
, ydest
); 
1323     CalcBoundingBox(xdest 
+ width
, ydest 
+ height
); 
1325     /* scale/translate size and position */ 
1326     wxCoord xx 
= XLOG2DEV(xdest
); 
1327     wxCoord yy 
= YLOG2DEV(ydest
); 
1329     if ( source
->m_isMemDC 
) 
1331         wxMemoryDC 
*memDC 
= (wxMemoryDC
*) source
; 
1332         DoDrawSubBitmap(memDC
->GetSelectedObject(), 
1333                         xsrc
, ysrc
, width
, height
, 
1334                         xdest
, ydest
, rop
, useMask
); 
1338         wxCoord ww 
= XLOG2DEVREL(width
); 
1339         wxCoord hh 
= YLOG2DEVREL(height
); 
1341         m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1342         m_MGLDC
->bitBlt(*source
->GetMGLDC(), 
1343                         xsrc
, ysrc
, xsrc 
+ ww
, ysrc 
+ hh
, 
1344                         xx
, yy
, LogicalFunctionToMGLRop(rop
)); 
1350 void wxDC::DoDrawBitmap(const wxBitmap 
&bmp
, wxCoord x
, wxCoord y
, bool useMask
) 
1352     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1353     wxCHECK_RET( bmp
.Ok(), wxT("invalid bitmap") ); 
1355     wxCoord w 
= bmp
.GetWidth(); 
1356     wxCoord h 
= bmp
.GetHeight(); 
1358     DoDrawSubBitmap(bmp
, 0, 0, w
, h
, x
, y
, m_logicalFunction
, useMask
); 
1361 void wxDC::DoDrawIcon(const wxIcon
& icon
, wxCoord x
, wxCoord y
) 
1363     // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why 
1364     DoDrawBitmap((const wxBitmap
&)icon
, x
, y
, true); 
1368 static inline void DoBitBlt(const wxBitmap
& src
, MGLDevCtx 
*dst
, 
1369                             int sx
, int sy
, int sw
, int sh
, 
1370                             int dx
, int dy
, int dw
, int dh
, 
1371                             int rop
, bool useStretching
, bool putSection
) 
1373     bitmap_t 
*bmp 
= src
.GetMGLbitmap_t(); 
1377             dst
->putBitmap(dx
, dy
, bmp
, rop
); 
1379             dst
->putBitmapSection(sx
, sy
, sx 
+ sw
, sy 
+ sh
, dx
, dy
, bmp
, rop
); 
1384             dst
->stretchBitmap(dx
, dy
, dx 
+ dw
, dy 
+ dh
, bmp
, rop
); 
1386             dst
->stretchBitmapSection(sx
, sy
, sx 
+ sw
, sy 
+ sh
, 
1387                                       dx
, dy
, dx 
+ dw
, dy 
+ dh
, bmp
, rop
); 
1391 void wxDC::DoDrawSubBitmap(const wxBitmap 
&bmp
, 
1392                            wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
, 
1393                            wxCoord destx
, wxCoord desty
, int rop
, bool useMask
) 
1395     wxCHECK_RET( Ok(), wxT("invalid dc") ); 
1396     wxCHECK_RET( bmp
.Ok(), wxT("invalid bitmap") ); 
1398     CalcBoundingBox(x
, y
); 
1399     CalcBoundingBox(x 
+ w
, y 
+ h
); 
1401     wxCoord dx 
= XLOG2DEV(destx
); 
1402     wxCoord dy 
= YLOG2DEV(desty
); 
1403     wxCoord dw 
= XLOG2DEVREL(w
); 
1404     wxCoord dh 
= YLOG2DEVREL(h
); 
1406     m_MGLDC
->makeCurrent(); // will go away with MGL6.0 
1408     bool useStretching 
= ((w 
!= dw
) || (h 
!= dh
)); 
1409     bool putSection 
= (w 
!= bmp
.GetWidth() || h 
!= bmp
.GetHeight()); 
1410     MGL_writeModeType mglRop 
= (MGL_writeModeType
)LogicalFunctionToMGLRop(rop
); 
1412     if ( bmp
.GetDepth() == 1 ) 
1414         // Mono bitmaps are handled in special way -- all 1s are drawn in 
1415         // foreground colours, all 0s in background colour. 
1417         ((wxBitmap
&)bmp
).SetMonoPalette(m_textForegroundColour
, m_textBackgroundColour
); 
1420     if ( useMask 
&& bmp
.GetMask() ) 
1422         // Since MGL does not support masks directly (in MGL, mask is handled 
1423         // in same way as in wxImage, i.e. there is one "key" color), we 
1424         // simulate masked bitblt in 6 steps (same as in MSW): 
1426         // 1. Create a temporary bitmap and copy the destination area into it. 
1427         // 2. Copy the source area into the temporary bitmap using the 
1428         //    specified logical function. 
1429         // 3. Set the masked area in the temporary bitmap to BLACK by ANDing 
1430         //    the mask bitmap with the temp bitmap with the foreground colour 
1431         //    set to WHITE and the bg colour set to BLACK. 
1432         // 4. Set the unmasked area in the destination area to BLACK by 
1433         //    ANDing the mask bitmap with the destination area with the 
1434         //    foreground colour set to BLACK and the background colour set 
1436         // 5. OR the temporary bitmap with the destination area. 
1437         // 6. Delete the temporary bitmap. 
1439         // This sequence of operations ensures that the source's transparent 
1440         // area need not be black, and logical functions are supported. 
1442         wxBitmap mask 
= bmp
.GetMask()->GetBitmap(); 
1446         if ( GetDepth() <= 8 ) 
1448             temp 
= new MGLMemoryDC(dw
, dh
, GetDepth(), NULL
); 
1450             tempdc
.SetMGLDC(temp
, false); 
1451             tempdc
.SetPalette(m_palette
); 
1456             m_MGLDC
->getPixelFormat(pf
); 
1457             temp 
= new MGLMemoryDC(dw
, dh
, GetDepth(), &pf
); 
1460         wxCHECK_RET( temp
->isValid(), wxT("cannot create temporary dc") ); 
1462         temp
->bitBlt(*m_MGLDC
, dx
, dy
, dx 
+ dw
, dy 
+ dh
, 0, 0, MGL_REPLACE_MODE
); 
1464         DoBitBlt(bmp
, temp
, x
, y
, w
, h
, 0, 0, dw
, dh
, mglRop
, 
1465                  useStretching
, putSection
); 
1467         mask
.SetMonoPalette(wxColour(0,0,0), wxColour(255,255,255)); 
1468         DoBitBlt(mask
, temp
, x
, y
, w
, h
, 0, 0, dw
, dh
, MGL_R2_MASKSRC
, 
1469                  useStretching
, putSection
); 
1470         DoBitBlt(mask
, m_MGLDC
, x
, y
, w
, h
, dx
, dy
, dw
, dh
, MGL_R2_MASKNOTSRC
, 
1471                  useStretching
, putSection
); 
1473         m_MGLDC
->bitBlt(*temp
, 0, 0, dw
, dh
, dx
, dy
, MGL_OR_MODE
); 
1480         DoBitBlt(bmp
, m_MGLDC
, x
, y
, w
, h
, dx
, dy
, dw
, dh
, mglRop
, 
1481                  useStretching
, putSection
);