]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/dc.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // =========================================================================== 
  14 // =========================================================================== 
  16 // --------------------------------------------------------------------------- 
  18 // --------------------------------------------------------------------------- 
  21     #pragma implementation "dc.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  32     #include "wx/window.h" 
  35     #include "wx/dialog.h" 
  37     #include "wx/bitmap.h" 
  38     #include "wx/dcmemory.h" 
  43 #include "wx/sysopt.h" 
  44 #include "wx/dcprint.h" 
  45 #include "wx/module.h" 
  50 #include "wx/msw/private.h" // needs to be before #include <commdlg.h> 
  52 #if wxUSE_COMMON_DIALOGS 
  60 IMPLEMENT_ABSTRACT_CLASS(wxDC
, wxDCBase
) 
  62 // --------------------------------------------------------------------------- 
  64 // --------------------------------------------------------------------------- 
  66 static const int VIEWPORT_EXTENT 
= 1000; 
  68 static const int MM_POINTS 
= 9; 
  69 static const int MM_METRIC 
= 10; 
  71 // usually this is defined in math.h 
  73     static const double M_PI 
= 3.14159265358979323846; 
  76 // ROPs which don't have standard names (see "Ternary Raster Operations" in the 
  77 // MSDN docs for how this and other numbers in wxDC::Blit() are obtained) 
  78 #define DSTCOPY 0x00AA0029      // a.k.a. NOP operation 
  80 // --------------------------------------------------------------------------- 
  82 // --------------------------------------------------------------------------- 
  84 // convert degrees to radians 
  85 static inline double DegToRad(double deg
) { return (deg 
* M_PI
) / 180.0; } 
  87 // ---------------------------------------------------------------------------- 
  89 // ---------------------------------------------------------------------------- 
  91 // instead of duplicating the same code which sets and then restores text 
  92 // colours in each wxDC method working with wxSTIPPLE_MASK_OPAQUE brushes, 
  93 // encapsulate this in a small helper class 
  95 // wxColourChanger: changes the text colours in the ctor if required and 
  96 //                  restores them in the dtor 
 100     wxColourChanger(wxDC
& dc
); 
 106     COLORREF m_colFgOld
, m_colBgOld
; 
 111 // =========================================================================== 
 113 // =========================================================================== 
 115 // ---------------------------------------------------------------------------- 
 117 // ---------------------------------------------------------------------------- 
 119 wxColourChanger::wxColourChanger(wxDC
& dc
) : m_dc(dc
) 
 121     if ( dc
.GetBrush().GetStyle() == wxSTIPPLE_MASK_OPAQUE 
) 
 123         HDC hdc 
= GetHdcOf(dc
); 
 124         m_colFgOld 
= ::GetTextColor(hdc
); 
 125         m_colBgOld 
= ::GetBkColor(hdc
); 
 127         // note that Windows convention is opposite to wxWindows one, this is 
 128         // why text colour becomes the background one and vice versa 
 129         const wxColour
& colFg 
= dc
.GetTextForeground(); 
 132             ::SetBkColor(hdc
, colFg
.GetPixel()); 
 135         const wxColour
& colBg 
= dc
.GetTextBackground(); 
 138             ::SetTextColor(hdc
, colBg
.GetPixel()); 
 142                   dc
.GetBackgroundMode() == wxTRANSPARENT 
? TRANSPARENT
 
 145         // flag which telsl us to undo changes in the dtor 
 150         // nothing done, nothing to undo 
 155 wxColourChanger::~wxColourChanger() 
 159         // restore the colours we changed 
 160         HDC hdc 
= GetHdcOf(m_dc
); 
 162         ::SetBkMode(hdc
, TRANSPARENT
); 
 163         ::SetTextColor(hdc
, m_colFgOld
); 
 164         ::SetBkColor(hdc
, m_colBgOld
); 
 168 // --------------------------------------------------------------------------- 
 170 // --------------------------------------------------------------------------- 
 172 // Default constructor 
 183 #endif // wxUSE_PALETTE 
 188     m_windowExtX 
= VIEWPORT_EXTENT
; 
 189     m_windowExtY 
= VIEWPORT_EXTENT
; 
 197         SelectOldObjects(m_hDC
); 
 199         // if we own the HDC, we delete it, otherwise we just release it 
 203             ::DeleteDC(GetHdc()); 
 205         else // we don't own our HDC 
 209                 ::ReleaseDC(GetHwndOf(m_canvas
), GetHdc()); 
 213                 // Must have been a wxScreenDC 
 214                 ::ReleaseDC((HWND
) NULL
, GetHdc()); 
 220 // This will select current objects out of the DC, 
 221 // which is what you have to do before deleting the 
 223 void wxDC::SelectOldObjects(WXHDC dc
) 
 229             ::SelectObject((HDC
) dc
, (HBITMAP
) m_oldBitmap
); 
 230             if (m_selectedBitmap
.Ok()) 
 232                 m_selectedBitmap
.SetSelectedInto(NULL
); 
 238             ::SelectObject((HDC
) dc
, (HPEN
) m_oldPen
); 
 243             ::SelectObject((HDC
) dc
, (HBRUSH
) m_oldBrush
); 
 248             ::SelectObject((HDC
) dc
, (HFONT
) m_oldFont
); 
 255             ::SelectPalette((HDC
) dc
, (HPALETTE
) m_oldPalette
, FALSE
); 
 258 #endif // wxUSE_PALETTE 
 261     m_brush 
= wxNullBrush
; 
 264     m_palette 
= wxNullPalette
; 
 265 #endif // wxUSE_PALETTE 
 267     m_backgroundBrush 
= wxNullBrush
; 
 268     m_selectedBitmap 
= wxNullBitmap
; 
 271 // --------------------------------------------------------------------------- 
 273 // --------------------------------------------------------------------------- 
 275 void wxDC::UpdateClipBox() 
 277 #ifdef __WXMICROWIN__ 
 278     if (!GetHDC()) return; 
 282     GetClipBox(GetHdc(), &rect
); 
 284     m_clipX1 
= (wxCoord
) XDEV2LOG(rect
.left
); 
 285     m_clipY1 
= (wxCoord
) YDEV2LOG(rect
.top
); 
 286     m_clipX2 
= (wxCoord
) XDEV2LOG(rect
.right
); 
 287     m_clipY2 
= (wxCoord
) YDEV2LOG(rect
.bottom
); 
 290 void wxDC::DoSetClippingRegion(wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
) 
 292 #ifdef __WXMICROWIN__ 
 293     if (!GetHDC()) return; 
 298     // the region coords are always the device ones, so do the translation 
 301     // FIXME: possible +/-1 error here, to check! 
 302     HRGN hrgn 
= ::CreateRectRgn(LogicalToDeviceX(x
), 
 304                                 LogicalToDeviceX(x 
+ w
), 
 305                                 LogicalToDeviceY(y 
+ h
)); 
 308         wxLogLastError(_T("CreateRectRgn")); 
 312         if ( ::SelectClipRgn(GetHdc(), hrgn
) == ERROR 
) 
 314             wxLogLastError(_T("SelectClipRgn")); 
 322 void wxDC::DoSetClippingRegionAsRegion(const wxRegion
& region
) 
 324 #ifdef __WXMICROWIN__ 
 325     if (!GetHDC()) return; 
 328     wxCHECK_RET( GetHrgnOf(region
), wxT("invalid clipping region") ); 
 333     SelectClipRgn(GetHdc(), GetHrgnOf(region
)); 
 335     ExtSelectClipRgn(GetHdc(), GetHrgnOf(region
), RGN_AND
); 
 341 void wxDC::DestroyClippingRegion() 
 343 #ifdef __WXMICROWIN__ 
 344     if (!GetHDC()) return; 
 347     if (m_clipping 
&& m_hDC
) 
 349         // TODO: this should restore the previous clipping region, 
 350         // so that OnPaint processing works correctly, and the update clipping region 
 351         // doesn't get destroyed after the first DestroyClippingRegion. 
 352         HRGN rgn 
= CreateRectRgn(0, 0, 32000, 32000); 
 353         SelectClipRgn(GetHdc(), rgn
); 
 360 // --------------------------------------------------------------------------- 
 361 // query capabilities 
 362 // --------------------------------------------------------------------------- 
 364 bool wxDC::CanDrawBitmap() const 
 369 bool wxDC::CanGetTextExtent() const 
 371 #ifdef __WXMICROWIN__ 
 372     // TODO Extend MicroWindows' GetDeviceCaps function 
 375     // What sort of display is it? 
 376     int technology 
= ::GetDeviceCaps(GetHdc(), TECHNOLOGY
); 
 378     return (technology 
== DT_RASDISPLAY
) || (technology 
== DT_RASPRINTER
); 
 382 int wxDC::GetDepth() const 
 384 #ifdef __WXMICROWIN__ 
 385     if (!GetHDC()) return 16; 
 388     return (int)::GetDeviceCaps(GetHdc(), BITSPIXEL
); 
 391 // --------------------------------------------------------------------------- 
 393 // --------------------------------------------------------------------------- 
 397 #ifdef __WXMICROWIN__ 
 398     if (!GetHDC()) return; 
 404         GetClientRect((HWND
) m_canvas
->GetHWND(), &rect
); 
 408         // No, I think we should simply ignore this if printing on e.g. 
 410         // wxCHECK_RET( m_selectedBitmap.Ok(), wxT("this DC can't be cleared") ); 
 411         if (!m_selectedBitmap
.Ok()) 
 414         rect
.left 
= 0; rect
.top 
= 0; 
 415         rect
.right 
= m_selectedBitmap
.GetWidth(); 
 416         rect
.bottom 
= m_selectedBitmap
.GetHeight(); 
 419     (void) ::SetMapMode(GetHdc(), MM_TEXT
); 
 421     DWORD colour 
= GetBkColor(GetHdc()); 
 422     HBRUSH brush 
= CreateSolidBrush(colour
); 
 423     FillRect(GetHdc(), &rect
, brush
); 
 426     ::SetMapMode(GetHdc(), MM_ANISOTROPIC
); 
 427     ::SetViewportExtEx(GetHdc(), VIEWPORT_EXTENT
, VIEWPORT_EXTENT
, NULL
); 
 428     ::SetWindowExtEx(GetHdc(), m_windowExtX
, m_windowExtY
, NULL
); 
 429     ::SetViewportOrgEx(GetHdc(), (int)m_deviceOriginX
, (int)m_deviceOriginY
, NULL
); 
 430     ::SetWindowOrgEx(GetHdc(), (int)m_logicalOriginX
, (int)m_logicalOriginY
, NULL
); 
 433 void wxDC::DoFloodFill(wxCoord x
, wxCoord y
, const wxColour
& col
, int style
) 
 435 #ifdef __WXMICROWIN__ 
 436     if (!GetHDC()) return; 
 439     if ( !::ExtFloodFill(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), 
 441                          style 
== wxFLOOD_SURFACE 
? FLOODFILLSURFACE
 
 444         // quoting from the MSDN docs: 
 446         //      Following are some of the reasons this function might fail: 
 448         //      * The filling could not be completed. 
 449         //      * The specified point has the boundary color specified by the 
 450         //        crColor parameter (if FLOODFILLBORDER was requested). 
 451         //      * The specified point does not have the color specified by 
 452         //        crColor (if FLOODFILLSURFACE was requested) 
 453         //      * The point is outside the clipping region that is, it is not 
 454         //        visible on the device. 
 456         wxLogLastError(wxT("ExtFloodFill")); 
 459     CalcBoundingBox(x
, y
); 
 462 bool wxDC::DoGetPixel(wxCoord x
, wxCoord y
, wxColour 
*col
) const 
 464 #ifdef __WXMICROWIN__ 
 465     if (!GetHDC()) return FALSE
; 
 468     wxCHECK_MSG( col
, FALSE
, _T("NULL colour parameter in wxDC::GetPixel") ); 
 470     // get the color of the pixel 
 471     COLORREF pixelcolor 
= ::GetPixel(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
)); 
 473     wxRGBToColour(*col
, pixelcolor
); 
 478 void wxDC::DoCrossHair(wxCoord x
, wxCoord y
) 
 480 #ifdef __WXMICROWIN__ 
 481     if (!GetHDC()) return; 
 484     wxCoord x1 
= x
-VIEWPORT_EXTENT
; 
 485     wxCoord y1 
= y
-VIEWPORT_EXTENT
; 
 486     wxCoord x2 
= x
+VIEWPORT_EXTENT
; 
 487     wxCoord y2 
= y
+VIEWPORT_EXTENT
; 
 489     (void)MoveToEx(GetHdc(), XLOG2DEV(x1
), YLOG2DEV(y
), NULL
); 
 490     (void)LineTo(GetHdc(), XLOG2DEV(x2
), YLOG2DEV(y
)); 
 492     (void)MoveToEx(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y1
), NULL
); 
 493     (void)LineTo(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y2
)); 
 495     CalcBoundingBox(x1
, y1
); 
 496     CalcBoundingBox(x2
, y2
); 
 499 void wxDC::DoDrawLine(wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
) 
 501 #ifdef __WXMICROWIN__ 
 502     if (!GetHDC()) return; 
 505     (void)MoveToEx(GetHdc(), XLOG2DEV(x1
), YLOG2DEV(y1
), NULL
); 
 506     (void)LineTo(GetHdc(), XLOG2DEV(x2
), YLOG2DEV(y2
)); 
 508     // Normalization: Windows doesn't draw the last point of the line. 
 509     // But apparently neither does GTK+, so we take it out again. 
 510 //    (void)LineTo(GetHdc(), XLOG2DEV(x2) + 1, YLOG2DEV(y2)); 
 512     CalcBoundingBox(x1
, y1
); 
 513     CalcBoundingBox(x2
, y2
); 
 516 // Draws an arc of a circle, centred on (xc, yc), with starting point (x1, y1) 
 517 // and ending at (x2, y2) 
 518 void wxDC::DoDrawArc(wxCoord x1
, wxCoord y1
, 
 519                      wxCoord x2
, wxCoord y2
, 
 520                      wxCoord xc
, wxCoord yc
) 
 522 #ifdef __WXMICROWIN__ 
 523     if (!GetHDC()) return; 
 526     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 530     double radius 
= (double)sqrt(dx
*dx
+dy
*dy
); 
 531     wxCoord r 
= (wxCoord
)radius
; 
 533     // treat the special case of full circle separately 
 534     if ( x1 
== x2 
&& y1 
== y2 
) 
 536         DrawEllipse(xc 
- r
, yc 
- r
, 2*r
, 2*r
); 
 540     wxCoord xx1 
= XLOG2DEV(x1
); 
 541     wxCoord yy1 
= YLOG2DEV(y1
); 
 542     wxCoord xx2 
= XLOG2DEV(x2
); 
 543     wxCoord yy2 
= YLOG2DEV(y2
); 
 544     wxCoord xxc 
= XLOG2DEV(xc
); 
 545     wxCoord yyc 
= YLOG2DEV(yc
); 
 546     wxCoord ray 
= (wxCoord
) sqrt(double((xxc
-xx1
)*(xxc
-xx1
)+(yyc
-yy1
)*(yyc
-yy1
))); 
 548     wxCoord xxx1 
= (wxCoord
) (xxc
-ray
); 
 549     wxCoord yyy1 
= (wxCoord
) (yyc
-ray
); 
 550     wxCoord xxx2 
= (wxCoord
) (xxc
+ray
); 
 551     wxCoord yyy2 
= (wxCoord
) (yyc
+ray
); 
 553     if ( m_brush
.Ok() && m_brush
.GetStyle() != wxTRANSPARENT 
) 
 555         // Have to add 1 to bottom-right corner of rectangle 
 556         // to make semi-circles look right (crooked line otherwise). 
 557         // Unfortunately this is not a reliable method, depends 
 558         // on the size of shape. 
 559         // TODO: figure out why this happens! 
 560         Pie(GetHdc(),xxx1
,yyy1
,xxx2
+1,yyy2
+1, xx1
,yy1
,xx2
,yy2
); 
 564         Arc(GetHdc(),xxx1
,yyy1
,xxx2
,yyy2
, xx1
,yy1
,xx2
,yy2
); 
 567     CalcBoundingBox(xc 
- r
, yc 
- r
); 
 568     CalcBoundingBox(xc 
+ r
, yc 
+ r
); 
 571 void wxDC::DoDrawCheckMark(wxCoord x1
, wxCoord y1
, 
 572                            wxCoord width
, wxCoord height
) 
 574 #ifdef __WXMICROWIN__ 
 575     if (!GetHDC()) return; 
 578     wxCoord x2 
= x1 
+ width
, 
 581 #if defined(__WIN32__) && !defined(__SC__) && !defined(__WXMICROWIN__) 
 588     DrawFrameControl(GetHdc(), &rect
, DFC_MENU
, DFCS_MENUCHECK
); 
 590     // In WIN16, draw a cross 
 591     HPEN blackPen 
= ::CreatePen(PS_SOLID
, 1, RGB(0, 0, 0)); 
 592     HPEN whiteBrush 
= (HPEN
)::GetStockObject(WHITE_BRUSH
); 
 593     HPEN hPenOld 
= (HPEN
)::SelectObject(GetHdc(), blackPen
); 
 594     HPEN hBrushOld 
= (HPEN
)::SelectObject(GetHdc(), whiteBrush
); 
 595     ::SetROP2(GetHdc(), R2_COPYPEN
); 
 596     Rectangle(GetHdc(), x1
, y1
, x2
, y2
); 
 597     MoveToEx(GetHdc(), x1
, y1
, NULL
); 
 598     LineTo(GetHdc(), x2
, y2
); 
 599     MoveToEx(GetHdc(), x2
, y1
, NULL
); 
 600     LineTo(GetHdc(), x1
, y2
); 
 601     ::SelectObject(GetHdc(), hPenOld
); 
 602     ::SelectObject(GetHdc(), hBrushOld
); 
 603     ::DeleteObject(blackPen
); 
 606     CalcBoundingBox(x1
, y1
); 
 607     CalcBoundingBox(x2
, y2
); 
 610 void wxDC::DoDrawPoint(wxCoord x
, wxCoord y
) 
 612 #ifdef __WXMICROWIN__ 
 613     if (!GetHDC()) return; 
 616     COLORREF color 
= 0x00ffffff; 
 619         color 
= m_pen
.GetColour().GetPixel(); 
 622     SetPixel(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), color
); 
 624     CalcBoundingBox(x
, y
); 
 627 void wxDC::DoDrawPolygon(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
,int fillStyle
) 
 629 #ifdef __WXMICROWIN__ 
 630     if (!GetHDC()) return; 
 633     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 635     // Do things less efficiently if we have offsets 
 636     if (xoffset 
!= 0 || yoffset 
!= 0) 
 638         POINT 
*cpoints 
= new POINT
[n
]; 
 640         for (i 
= 0; i 
< n
; i
++) 
 642             cpoints
[i
].x 
= (int)(points
[i
].x 
+ xoffset
); 
 643             cpoints
[i
].y 
= (int)(points
[i
].y 
+ yoffset
); 
 645             CalcBoundingBox(cpoints
[i
].x
, cpoints
[i
].y
); 
 647         int prev 
= SetPolyFillMode(GetHdc(),fillStyle
==wxODDEVEN_RULE
?ALTERNATE
:WINDING
); 
 648         (void)Polygon(GetHdc(), cpoints
, n
); 
 649         SetPolyFillMode(GetHdc(),prev
); 
 655         for (i 
= 0; i 
< n
; i
++) 
 656             CalcBoundingBox(points
[i
].x
, points
[i
].y
); 
 658         int prev 
= SetPolyFillMode(GetHdc(),fillStyle
==wxODDEVEN_RULE
?ALTERNATE
:WINDING
); 
 659         (void)Polygon(GetHdc(), (POINT
*) points
, n
); 
 660         SetPolyFillMode(GetHdc(),prev
); 
 664 void wxDC::DoDrawLines(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
) 
 666 #ifdef __WXMICROWIN__ 
 667     if (!GetHDC()) return; 
 670     // Do things less efficiently if we have offsets 
 671     if (xoffset 
!= 0 || yoffset 
!= 0) 
 673         POINT 
*cpoints 
= new POINT
[n
]; 
 675         for (i 
= 0; i 
< n
; i
++) 
 677             cpoints
[i
].x 
= (int)(points
[i
].x 
+ xoffset
); 
 678             cpoints
[i
].y 
= (int)(points
[i
].y 
+ yoffset
); 
 680             CalcBoundingBox(cpoints
[i
].x
, cpoints
[i
].y
); 
 682         (void)Polyline(GetHdc(), cpoints
, n
); 
 688         for (i 
= 0; i 
< n
; i
++) 
 689             CalcBoundingBox(points
[i
].x
, points
[i
].y
); 
 691         (void)Polyline(GetHdc(), (POINT
*) points
, n
); 
 695 void wxDC::DoDrawRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
) 
 697 #ifdef __WXMICROWIN__ 
 698     if (!GetHDC()) return; 
 701     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 703     wxCoord x2 
= x 
+ width
; 
 704     wxCoord y2 
= y 
+ height
; 
 706     if ((m_logicalFunction 
== wxCOPY
) && (m_pen
.GetStyle() == wxTRANSPARENT
)) 
 709         rect
.left 
= XLOG2DEV(x
); 
 710         rect
.top 
= YLOG2DEV(y
); 
 711         rect
.right 
= XLOG2DEV(x2
); 
 712         rect
.bottom 
= YLOG2DEV(y2
); 
 713         (void)FillRect(GetHdc(), &rect
, (HBRUSH
)m_brush
.GetResourceHandle() ); 
 717         // Windows draws the filled rectangles without outline (i.e. drawn with a 
 718         // transparent pen) one pixel smaller in both directions and we want them 
 719         // to have the same size regardless of which pen is used - adjust 
 721         // I wonder if this shouldn´t be done after the LOG2DEV() conversions. RR. 
 722         if ( m_pen
.GetStyle() == wxTRANSPARENT 
) 
 728         (void)Rectangle(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
)); 
 732     CalcBoundingBox(x
, y
); 
 733     CalcBoundingBox(x2
, y2
); 
 736 void wxDC::DoDrawRoundedRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double radius
) 
 738 #ifdef __WXMICROWIN__ 
 739     if (!GetHDC()) return; 
 742     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 744     // Now, a negative radius value is interpreted to mean 
 745     // 'the proportion of the smallest X or Y dimension' 
 749         double smallest 
= 0.0; 
 754         radius 
= (- radius 
* smallest
); 
 757     wxCoord x2 
= (x
+width
); 
 758     wxCoord y2 
= (y
+height
); 
 760     // Windows draws the filled rectangles without outline (i.e. drawn with a 
 761     // transparent pen) one pixel smaller in both directions and we want them 
 762     // to have the same size regardless of which pen is used - adjust 
 763     if ( m_pen
.GetStyle() == wxTRANSPARENT 
) 
 769     (void)RoundRect(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), 
 770         YLOG2DEV(y2
), (int) (2*XLOG2DEV(radius
)), (int)( 2*YLOG2DEV(radius
))); 
 772     CalcBoundingBox(x
, y
); 
 773     CalcBoundingBox(x2
, y2
); 
 776 void wxDC::DoDrawEllipse(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
) 
 778 #ifdef __WXMICROWIN__ 
 779     if (!GetHDC()) return; 
 782     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 784     wxCoord x2 
= (x
+width
); 
 785     wxCoord y2 
= (y
+height
); 
 787     (void)Ellipse(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
)); 
 789     CalcBoundingBox(x
, y
); 
 790     CalcBoundingBox(x2
, y2
); 
 793 // Chris Breeze 20/5/98: first implementation of DrawEllipticArc on Windows 
 794 void wxDC::DoDrawEllipticArc(wxCoord x
,wxCoord y
,wxCoord w
,wxCoord h
,double sa
,double ea
) 
 796 #ifdef __WXMICROWIN__ 
 797     if (!GetHDC()) return; 
 800     wxColourChanger 
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling 
 805     int rx1 
= XLOG2DEV(x
+w
/2); 
 806     int ry1 
= YLOG2DEV(y
+h
/2); 
 813     rx1 
+= (int)(100.0 * abs(w
) * cos(sa
)); 
 814     ry1 
-= (int)(100.0 * abs(h
) * m_signY 
* sin(sa
)); 
 815     rx2 
+= (int)(100.0 * abs(w
) * cos(ea
)); 
 816     ry2 
-= (int)(100.0 * abs(h
) * m_signY 
* sin(ea
)); 
 818     // draw pie with NULL_PEN first and then outline otherwise a line is 
 819     // drawn from the start and end points to the centre 
 820     HPEN hpenOld 
= (HPEN
) ::SelectObject(GetHdc(), (HPEN
) ::GetStockObject(NULL_PEN
)); 
 823         (void)Pie(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
)+1, YLOG2DEV(y2
)+1, 
 828         (void)Pie(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
)-1, XLOG2DEV(x2
)+1, YLOG2DEV(y2
), 
 829                   rx1
, ry1
-1, rx2
, ry2
-1); 
 832     ::SelectObject(GetHdc(), hpenOld
); 
 834     (void)Arc(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
), 
 837     CalcBoundingBox(x
, y
); 
 838     CalcBoundingBox(x2
, y2
); 
 841 void wxDC::DoDrawIcon(const wxIcon
& icon
, wxCoord x
, wxCoord y
) 
 843 #ifdef __WXMICROWIN__ 
 844     if (!GetHDC()) return; 
 847     wxCHECK_RET( icon
.Ok(), wxT("invalid icon in DrawIcon") ); 
 850     ::DrawIconEx(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), GetHiconOf(icon
), icon
.GetWidth(), icon
.GetHeight(), 0, NULL
, DI_NORMAL
); 
 852     ::DrawIcon(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), GetHiconOf(icon
)); 
 855     CalcBoundingBox(x
, y
); 
 856     CalcBoundingBox(x 
+ icon
.GetWidth(), y 
+ icon
.GetHeight()); 
 859 void wxDC::DoDrawBitmap( const wxBitmap 
&bmp
, wxCoord x
, wxCoord y
, bool useMask 
) 
 861 #ifdef __WXMICROWIN__ 
 862     if (!GetHDC()) return; 
 865     wxCHECK_RET( bmp
.Ok(), _T("invalid bitmap in wxDC::DrawBitmap") ); 
 867     int width 
= bmp
.GetWidth(), 
 868         height 
= bmp
.GetHeight(); 
 870     HBITMAP hbmpMask 
= 0; 
 875         wxMask 
*mask 
= bmp
.GetMask(); 
 877             hbmpMask 
= (HBITMAP
)mask
->GetMaskBitmap(); 
 881             // don't give assert here because this would break existing 
 882             // programs - just silently ignore useMask parameter 
 890         // use MaskBlt() with ROP which doesn't do anything to dst in the mask 
 892         // On some systems, MaskBlt succeeds yet is much much slower 
 893         // than the wxWindows fall-back implementation. So we need 
 894         // to be able to switch this on and off at runtime. 
 896 #if wxUSE_SYSTEM_OPTIONS 
 897         if (wxSystemOptions::GetOptionInt(wxT("no-maskblt")) == 0) 
 901             HDC hdcMem 
= ::CreateCompatibleDC(GetHdc()); 
 902             ::SelectObject(hdcMem
, GetHbitmapOf(bmp
)); 
 903             if (bmp
.GetPalette() && (::GetDeviceCaps(cdc
,BITSPIXEL
) <= 8)) { 
 904                 oldPal 
= ::SelectPalette( hdcMem
, (HPALETTE
)bmp
.GetPalette()->GetHPALETTE(), FALSE
); 
 905                 ::RealizePalette(hdcMem
); 
 907             ok 
= ::MaskBlt(cdc
, x
, y
, width
, height
, 
 910                             MAKEROP4(SRCCOPY
, DSTCOPY
)) != 0; 
 912                 ::SelectPalette(hdcMem
, oldPal
, FALSE
); 
 919             // Rather than reproduce wxDC::Blit, let's do it at the wxWin API 
 922             memDC
.SelectObject(bmp
); 
 924             Blit(x
, y
, width
, height
, &memDC
, 0, 0, wxCOPY
, useMask
); 
 926             memDC
.SelectObject(wxNullBitmap
); 
 929     else // no mask, just use BitBlt() 
 932         HDC memdc 
= ::CreateCompatibleDC( cdc 
); 
 933         HBITMAP hbitmap 
= (HBITMAP
) bmp
.GetHBITMAP( ); 
 935         wxASSERT_MSG( hbitmap
, wxT("bitmap is ok but HBITMAP is NULL?") ); 
 937         COLORREF old_textground 
= ::GetTextColor(GetHdc()); 
 938         COLORREF old_background 
= ::GetBkColor(GetHdc()); 
 939         if (m_textForegroundColour
.Ok()) 
 941             ::SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel() ); 
 943         if (m_textBackgroundColour
.Ok()) 
 945             ::SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() ); 
 948         if (bmp
.GetPalette() && (::GetDeviceCaps(cdc
,BITSPIXEL
) <= 8)) { 
 949             oldPal 
= ::SelectPalette( memdc
, (HPALETTE
)bmp
.GetPalette()->GetHPALETTE(), FALSE
); 
 950             ::RealizePalette(memdc
); 
 952         ::SelectObject( memdc
, hbitmap 
); 
 953         ::BitBlt( cdc
, x
, y
, width
, height
, memdc
, 0, 0, SRCCOPY
); 
 955             ::SelectPalette(memdc
, oldPal
, FALSE
); 
 958         ::SetTextColor(GetHdc(), old_textground
); 
 959         ::SetBkColor(GetHdc(), old_background
); 
 963 void wxDC::DoDrawText(const wxString
& text
, wxCoord x
, wxCoord y
) 
 965 #ifdef __WXMICROWIN__ 
 966     if (!GetHDC()) return; 
 969     DrawAnyText(text
, x
, y
); 
 971     // update the bounding box 
 972     CalcBoundingBox(x
, y
); 
 975     GetTextExtent(text
, &w
, &h
); 
 976     CalcBoundingBox(x 
+ w
, y 
+ h
); 
 979 void wxDC::DrawAnyText(const wxString
& text
, wxCoord x
, wxCoord y
) 
 981 #ifdef __WXMICROWIN__ 
 982     if (!GetHDC()) return; 
 985     // prepare for drawing the text 
 986     if ( m_textForegroundColour
.Ok() ) 
 987         SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel()); 
 989     DWORD old_background 
= 0; 
 990     if ( m_textBackgroundColour
.Ok() ) 
 992         old_background 
= SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() ); 
 995     SetBkMode(GetHdc(), m_backgroundMode 
== wxTRANSPARENT 
? TRANSPARENT
 
 998     if ( ::TextOut(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), 
 999                    text
.c_str(), text
.length()) == 0 ) 
1001         wxLogLastError(wxT("TextOut")); 
1004     // restore the old parameters (text foreground colour may be left because 
1005     // it never is set to anything else, but background should remain 
1006     // transparent even if we just drew an opaque string) 
1007     if ( m_textBackgroundColour
.Ok() ) 
1008         (void)SetBkColor(GetHdc(), old_background
); 
1010     SetBkMode(GetHdc(), TRANSPARENT
); 
1013 void wxDC::DoDrawRotatedText(const wxString
& text
, 
1014                              wxCoord x
, wxCoord y
, 
1017 #ifdef __WXMICROWIN__ 
1018     if (!GetHDC()) return; 
1021     // we test that we have some font because otherwise we should still use the 
1022     // "else" part below to avoid that DrawRotatedText(angle = 180) and 
1023     // DrawRotatedText(angle = 0) use different fonts (we can't use the default 
1024     // font for drawing rotated fonts unfortunately) 
1025     if ( (angle 
== 0.0) && m_font
.Ok() ) 
1027         DoDrawText(text
, x
, y
); 
1029 #ifndef __WXMICROWIN__ 
1032         // NB: don't take DEFAULT_GUI_FONT (a.k.a. wxSYS_DEFAULT_GUI_FONT) 
1033         //     because it's not TrueType and so can't have non zero 
1034         //     orientation/escapement under Win9x 
1035         wxFont font 
= m_font
.Ok() ? m_font 
: *wxSWISS_FONT
; 
1036         HFONT hfont 
= (HFONT
)font
.GetResourceHandle(); 
1038         if ( ::GetObject(hfont
, sizeof(lf
), &lf
) == 0 ) 
1040             wxLogLastError(wxT("GetObject(hfont)")); 
1043         // GDI wants the angle in tenth of degree 
1044         long angle10 
= (long)(angle 
* 10); 
1045         lf
.lfEscapement 
= angle10
; 
1046         lf
. lfOrientation 
= angle10
; 
1048         hfont 
= ::CreateFontIndirect(&lf
); 
1051             wxLogLastError(wxT("CreateFont")); 
1055             HFONT hfontOld 
= (HFONT
)::SelectObject(GetHdc(), hfont
); 
1057             DrawAnyText(text
, x
, y
); 
1059             (void)::SelectObject(GetHdc(), hfontOld
); 
1060             (void)::DeleteObject(hfont
); 
1063         // call the bounding box by adding all four vertices of the rectangle 
1064         // containing the text to it (simpler and probably not slower than 
1065         // determining which of them is really topmost/leftmost/...) 
1067         GetTextExtent(text
, &w
, &h
); 
1069         double rad 
= DegToRad(angle
); 
1071         // "upper left" and "upper right" 
1072         CalcBoundingBox(x
, y
); 
1073         CalcBoundingBox(x 
+ w
*cos(rad
), y 
- h
*sin(rad
)); 
1075         // "bottom left" and "bottom right" 
1076         x 
+= (wxCoord
)(h
*sin(rad
)); 
1077         y 
+= (wxCoord
)(h
*cos(rad
)); 
1078         CalcBoundingBox(x
, y
); 
1079         CalcBoundingBox(x 
+ h
*sin(rad
), y 
+ h
*cos(rad
)); 
1084 // --------------------------------------------------------------------------- 
1086 // --------------------------------------------------------------------------- 
1090 void wxDC::SetPalette(const wxPalette
& palette
) 
1092 #ifdef __WXMICROWIN__ 
1093     if (!GetHDC()) return; 
1096     // Set the old object temporarily, in case the assignment deletes an object 
1097     // that's not yet selected out. 
1100         ::SelectPalette(GetHdc(), (HPALETTE
) m_oldPalette
, FALSE
); 
1104     m_palette 
= palette
; 
1106     if (!m_palette
.Ok()) 
1108         // Setting a NULL colourmap is a way of restoring 
1109         // the original colourmap 
1112             ::SelectPalette(GetHdc(), (HPALETTE
) m_oldPalette
, FALSE
); 
1119     if (m_palette
.Ok() && m_palette
.GetHPALETTE()) 
1121         HPALETTE oldPal 
= ::SelectPalette(GetHdc(), (HPALETTE
) m_palette
.GetHPALETTE(), FALSE
); 
1123             m_oldPalette 
= (WXHPALETTE
) oldPal
; 
1125         ::RealizePalette(GetHdc()); 
1129 #endif // wxUSE_PALETTE 
1131 void wxDC::SetFont(const wxFont
& the_font
) 
1133 #ifdef __WXMICROWIN__ 
1134     if (!GetHDC()) return; 
1137     // Set the old object temporarily, in case the assignment deletes an object 
1138     // that's not yet selected out. 
1141         ::SelectObject(GetHdc(), (HFONT
) m_oldFont
); 
1150             ::SelectObject(GetHdc(), (HFONT
) m_oldFont
); 
1154     if (m_font
.Ok() && m_font
.GetResourceHandle()) 
1156         HFONT f 
= (HFONT
) ::SelectObject(GetHdc(), (HFONT
) m_font
.GetResourceHandle()); 
1157         if (f 
== (HFONT
) NULL
) 
1159             wxLogDebug(wxT("::SelectObject failed in wxDC::SetFont.")); 
1162             m_oldFont 
= (WXHFONT
) f
; 
1166 void wxDC::SetPen(const wxPen
& pen
) 
1168 #ifdef __WXMICROWIN__ 
1169     if (!GetHDC()) return; 
1172     // Set the old object temporarily, in case the assignment deletes an object 
1173     // that's not yet selected out. 
1176         ::SelectObject(GetHdc(), (HPEN
) m_oldPen
); 
1185             ::SelectObject(GetHdc(), (HPEN
) m_oldPen
); 
1191         if (m_pen
.GetResourceHandle()) 
1193             HPEN p 
= (HPEN
) ::SelectObject(GetHdc(), (HPEN
)m_pen
.GetResourceHandle()); 
1195                 m_oldPen 
= (WXHPEN
) p
; 
1200 void wxDC::SetBrush(const wxBrush
& brush
) 
1202 #ifdef __WXMICROWIN__ 
1203     if (!GetHDC()) return; 
1206     // Set the old object temporarily, in case the assignment deletes an object 
1207     // that's not yet selected out. 
1210         ::SelectObject(GetHdc(), (HBRUSH
) m_oldBrush
); 
1219             ::SelectObject(GetHdc(), (HBRUSH
) m_oldBrush
); 
1225         // to make sure the brush is alligned with the logical coordinates 
1226         wxBitmap 
*stipple 
= m_brush
.GetStipple(); 
1227         if ( stipple 
&& stipple
->Ok() ) 
1230                 ::SetBrushOrgEx(GetHdc(), 
1231                             m_deviceOriginX 
% stipple
->GetWidth(), 
1232                             m_deviceOriginY 
% stipple
->GetHeight(), 
1233                             NULL
);  // don't need previous brush origin 
1235                 ::SetBrushOrg(GetHdc(), 
1236                             m_deviceOriginX 
% stipple
->GetWidth(), 
1237                             m_deviceOriginY 
% stipple
->GetHeight()); 
1241         if ( m_brush
.GetResourceHandle() ) 
1244             b 
= (HBRUSH
) ::SelectObject(GetHdc(), (HBRUSH
)m_brush
.GetResourceHandle()); 
1246                 m_oldBrush 
= (WXHBRUSH
) b
; 
1251 void wxDC::SetBackground(const wxBrush
& brush
) 
1253 #ifdef __WXMICROWIN__ 
1254     if (!GetHDC()) return; 
1257     m_backgroundBrush 
= brush
; 
1259     if (!m_backgroundBrush
.Ok()) 
1264         bool customColours 
= TRUE
; 
1265         // If we haven't specified wxUSER_COLOURS, don't allow the panel/dialog box to 
1266         // change background colours from the control-panel specified colours. 
1267         if (m_canvas
->IsKindOf(CLASSINFO(wxWindow
)) && ((m_canvas
->GetWindowStyleFlag() & wxUSER_COLOURS
) != wxUSER_COLOURS
)) 
1268             customColours 
= FALSE
; 
1272             if (m_backgroundBrush
.GetStyle()==wxTRANSPARENT
) 
1274                 m_canvas
->SetTransparent(TRUE
); 
1278                 // New behaviour, 10/2/99: setting the background brush of a DC 
1279                 // doesn't affect the window background colour. However, 
1280                 // I'm leaving in the transparency setting because it's needed by 
1281                 // various controls (e.g. wxStaticText) to determine whether to draw 
1282                 // transparently or not. TODO: maybe this should be a new function 
1283                 // wxWindow::SetTransparency(). Should that apply to the child itself, or the 
1285                 //        m_canvas->SetBackgroundColour(m_backgroundBrush.GetColour()); 
1286                 m_canvas
->SetTransparent(FALSE
); 
1290     COLORREF new_color 
= m_backgroundBrush
.GetColour().GetPixel(); 
1292         (void)SetBkColor(GetHdc(), new_color
); 
1296 void wxDC::SetBackgroundMode(int mode
) 
1298 #ifdef __WXMICROWIN__ 
1299     if (!GetHDC()) return; 
1302     m_backgroundMode 
= mode
; 
1304     // SetBackgroundColour now only refers to text background 
1305     // and m_backgroundMode is used there 
1308     if (m_backgroundMode == wxTRANSPARENT) 
1309         ::SetBkMode(GetHdc(), TRANSPARENT); 
1311         ::SetBkMode(GetHdc(), OPAQUE); 
1312         Last change:  AC   29 Jan 101    8:54 pm 
1316 void wxDC::SetLogicalFunction(int function
) 
1318 #ifdef __WXMICROWIN__ 
1319     if (!GetHDC()) return; 
1322     m_logicalFunction 
= function
; 
1327 void wxDC::SetRop(WXHDC dc
) 
1329     if ( !dc 
|| m_logicalFunction 
< 0 ) 
1334     switch (m_logicalFunction
) 
1336         case wxCLEAR
:        rop 
= R2_BLACK
;         break; 
1337         case wxXOR
:          rop 
= R2_XORPEN
;        break; 
1338         case wxINVERT
:       rop 
= R2_NOT
;           break; 
1339         case wxOR_REVERSE
:   rop 
= R2_MERGEPENNOT
;   break; 
1340         case wxAND_REVERSE
:  rop 
= R2_MASKPENNOT
;    break; 
1341         case wxCOPY
:         rop 
= R2_COPYPEN
;       break; 
1342         case wxAND
:          rop 
= R2_MASKPEN
;       break; 
1343         case wxAND_INVERT
:   rop 
= R2_MASKNOTPEN
;    break; 
1344         case wxNO_OP
:        rop 
= R2_NOP
;           break; 
1345         case wxNOR
:          rop 
= R2_NOTMERGEPEN
;   break; 
1346         case wxEQUIV
:        rop 
= R2_NOTXORPEN
;     break; 
1347         case wxSRC_INVERT
:   rop 
= R2_NOTCOPYPEN
;    break; 
1348         case wxOR_INVERT
:    rop 
= R2_MERGENOTPEN
;   break; 
1349         case wxNAND
:         rop 
= R2_NOTMASKPEN
;    break; 
1350         case wxOR
:           rop 
= R2_MERGEPEN
;      break; 
1351         case wxSET
:          rop 
= R2_WHITE
;         break; 
1354            wxFAIL_MSG( wxT("unsupported logical function") ); 
1358     SetROP2(GetHdc(), rop
); 
1361 bool wxDC::StartDoc(const wxString
& WXUNUSED(message
)) 
1363     // We might be previewing, so return TRUE to let it continue. 
1371 void wxDC::StartPage() 
1375 void wxDC::EndPage() 
1379 // --------------------------------------------------------------------------- 
1381 // --------------------------------------------------------------------------- 
1383 wxCoord 
wxDC::GetCharHeight() const 
1385 #ifdef __WXMICROWIN__ 
1386     if (!GetHDC()) return 0; 
1389     TEXTMETRIC lpTextMetric
; 
1391     GetTextMetrics(GetHdc(), &lpTextMetric
); 
1393     return YDEV2LOGREL(lpTextMetric
.tmHeight
); 
1396 wxCoord 
wxDC::GetCharWidth() const 
1398 #ifdef __WXMICROWIN__ 
1399     if (!GetHDC()) return 0; 
1402     TEXTMETRIC lpTextMetric
; 
1404     GetTextMetrics(GetHdc(), &lpTextMetric
); 
1406     return XDEV2LOGREL(lpTextMetric
.tmAveCharWidth
); 
1409 void wxDC::DoGetTextExtent(const wxString
& string
, wxCoord 
*x
, wxCoord 
*y
, 
1410                            wxCoord 
*descent
, wxCoord 
*externalLeading
, 
1413 #ifdef __WXMICROWIN__ 
1418         if (descent
) *descent 
= 0; 
1419         if (externalLeading
) *externalLeading 
= 0; 
1427         wxASSERT_MSG( font
->Ok(), _T("invalid font in wxDC::GetTextExtent") ); 
1429         hfontOld 
= (HFONT
)::SelectObject(GetHdc(), GetHfontOf(*font
)); 
1431     else // don't change the font 
1439     GetTextExtentPoint(GetHdc(), string
, string
.length(), &sizeRect
); 
1440     GetTextMetrics(GetHdc(), &tm
); 
1442     if (x
) *x 
= XDEV2LOGREL(sizeRect
.cx
); 
1443     if (y
) *y 
= YDEV2LOGREL(sizeRect
.cy
); 
1444     if (descent
) *descent 
= tm
.tmDescent
; 
1445     if (externalLeading
) *externalLeading 
= tm
.tmExternalLeading
; 
1449         ::SelectObject(GetHdc(), hfontOld
); 
1453 void wxDC::SetMapMode(int mode
) 
1455 #ifdef __WXMICROWIN__ 
1456     if (!GetHDC()) return; 
1459     m_mappingMode 
= mode
; 
1461     int pixel_width 
= 0; 
1462     int pixel_height 
= 0; 
1466     pixel_width 
= GetDeviceCaps(GetHdc(), HORZRES
); 
1467     pixel_height 
= GetDeviceCaps(GetHdc(), VERTRES
); 
1468     mm_width 
= GetDeviceCaps(GetHdc(), HORZSIZE
); 
1469     mm_height 
= GetDeviceCaps(GetHdc(), VERTSIZE
); 
1471     if ((pixel_width 
== 0) || (pixel_height 
== 0) || (mm_width 
== 0) || (mm_height 
== 0)) 
1476     double mm2pixelsX 
= pixel_width
/mm_width
; 
1477     double mm2pixelsY 
= pixel_height
/mm_height
; 
1483             m_logicalScaleX 
= (twips2mm 
* mm2pixelsX
); 
1484             m_logicalScaleY 
= (twips2mm 
* mm2pixelsY
); 
1489             m_logicalScaleX 
= (pt2mm 
* mm2pixelsX
); 
1490             m_logicalScaleY 
= (pt2mm 
* mm2pixelsY
); 
1495             m_logicalScaleX 
= mm2pixelsX
; 
1496             m_logicalScaleY 
= mm2pixelsY
; 
1501             m_logicalScaleX 
= (mm2pixelsX
/10.0); 
1502             m_logicalScaleY 
= (mm2pixelsY
/10.0); 
1508             m_logicalScaleX 
= 1.0; 
1509             m_logicalScaleY 
= 1.0; 
1514     if (::GetMapMode(GetHdc()) != MM_ANISOTROPIC
) 
1515         ::SetMapMode(GetHdc(), MM_ANISOTROPIC
); 
1517     SetViewportExtEx(GetHdc(), VIEWPORT_EXTENT
, VIEWPORT_EXTENT
, NULL
); 
1518     m_windowExtX 
= (int)MS_XDEV2LOGREL(VIEWPORT_EXTENT
); 
1519     m_windowExtY 
= (int)MS_YDEV2LOGREL(VIEWPORT_EXTENT
); 
1520     ::SetWindowExtEx(GetHdc(), m_windowExtX
, m_windowExtY
, NULL
); 
1521     ::SetViewportOrgEx(GetHdc(), (int)m_deviceOriginX
, (int)m_deviceOriginY
, NULL
); 
1522     ::SetWindowOrgEx(GetHdc(), (int)m_logicalOriginX
, (int)m_logicalOriginY
, NULL
); 
1525 void wxDC::SetUserScale(double x
, double y
) 
1527 #ifdef __WXMICROWIN__ 
1528     if (!GetHDC()) return; 
1534     SetMapMode(m_mappingMode
); 
1537 void wxDC::SetAxisOrientation(bool xLeftRight
, bool yBottomUp
) 
1539 #ifdef __WXMICROWIN__ 
1540     if (!GetHDC()) return; 
1543     m_signX 
= xLeftRight 
? 1 : -1; 
1544     m_signY 
= yBottomUp 
? -1 : 1; 
1546     SetMapMode(m_mappingMode
); 
1549 void wxDC::SetSystemScale(double x
, double y
) 
1551 #ifdef __WXMICROWIN__ 
1552     if (!GetHDC()) return; 
1558     SetMapMode(m_mappingMode
); 
1561 void wxDC::SetLogicalOrigin(wxCoord x
, wxCoord y
) 
1563 #ifdef __WXMICROWIN__ 
1564     if (!GetHDC()) return; 
1567     m_logicalOriginX 
= x
; 
1568     m_logicalOriginY 
= y
; 
1570     ::SetWindowOrgEx(GetHdc(), (int)m_logicalOriginX
, (int)m_logicalOriginY
, NULL
); 
1573 void wxDC::SetDeviceOrigin(wxCoord x
, wxCoord y
) 
1575 #ifdef __WXMICROWIN__ 
1576     if (!GetHDC()) return; 
1579     m_deviceOriginX 
= x
; 
1580     m_deviceOriginY 
= y
; 
1582     ::SetViewportOrgEx(GetHdc(), (int)m_deviceOriginX
, (int)m_deviceOriginY
, NULL
); 
1585 // --------------------------------------------------------------------------- 
1586 // coordinates transformations 
1587 // --------------------------------------------------------------------------- 
1589 wxCoord 
wxDCBase::DeviceToLogicalX(wxCoord x
) const 
1591     double xRel 
= x 
- m_deviceOriginX
; 
1592     xRel 
/= m_logicalScaleX
*m_userScaleX
*m_signX
*m_scaleX
; 
1593     return (wxCoord
)(xRel 
+ m_logicalOriginX
); 
1596 wxCoord 
wxDCBase::DeviceToLogicalXRel(wxCoord x
) const 
1598     return (wxCoord
) ((x
)/(m_logicalScaleX
*m_userScaleX
*m_signX
*m_scaleX
)); 
1601 wxCoord 
wxDCBase::DeviceToLogicalY(wxCoord y
) const 
1603     double yRel 
= y 
- m_deviceOriginY
; 
1604     yRel 
/= m_logicalScaleY
*m_userScaleY
*m_signY
*m_scaleY
; 
1605     return (wxCoord
)(yRel 
+ m_logicalOriginY
); 
1608 wxCoord 
wxDCBase::DeviceToLogicalYRel(wxCoord y
) const 
1610     return (wxCoord
) ((y
)/(m_logicalScaleY
*m_userScaleY
*m_signY
*m_scaleY
)); 
1613 wxCoord 
wxDCBase::LogicalToDeviceX(wxCoord x
) const 
1615     return (wxCoord
) ((x 
- m_logicalOriginX
)*m_logicalScaleX
*m_userScaleX
*m_signX
*m_scaleX 
+ m_deviceOriginX
); 
1618 wxCoord 
wxDCBase::LogicalToDeviceXRel(wxCoord x
) const 
1620     return (wxCoord
) (x
*m_logicalScaleX
*m_userScaleX
*m_signX
*m_scaleX
); 
1623 wxCoord 
wxDCBase::LogicalToDeviceY(wxCoord y
) const 
1625     return (wxCoord
) ((y 
- m_logicalOriginY
)*m_logicalScaleY
*m_userScaleY
*m_signY
*m_scaleY 
+ m_deviceOriginY
); 
1628 wxCoord 
wxDCBase::LogicalToDeviceYRel(wxCoord y
) const 
1630     return (wxCoord
) (y
*m_logicalScaleY
*m_userScaleY
*m_signY
*m_scaleY
); 
1633 // --------------------------------------------------------------------------- 
1635 // --------------------------------------------------------------------------- 
1637 bool wxDC::DoBlit(wxCoord xdest
, wxCoord ydest
, 
1638                   wxCoord width
, wxCoord height
, 
1639                   wxDC 
*source
, wxCoord xsrc
, wxCoord ysrc
, 
1640                   int rop
, bool useMask
, 
1641                   wxCoord xsrcMask
, wxCoord ysrcMask
) 
1643 #ifdef __WXMICROWIN__ 
1644     if (!GetHDC()) return FALSE
; 
1647     wxMask 
*mask 
= NULL
; 
1650         const wxBitmap
& bmp 
= source
->m_selectedBitmap
; 
1651         mask 
= bmp
.GetMask(); 
1653         if ( !(bmp
.Ok() && mask 
&& mask
->GetMaskBitmap()) ) 
1655             // don't give assert here because this would break existing 
1656             // programs - just silently ignore useMask parameter 
1661     if (xsrcMask 
== -1 && ysrcMask 
== -1) 
1663         xsrcMask 
= xsrc
; ysrcMask 
= ysrc
; 
1666     COLORREF old_textground 
= ::GetTextColor(GetHdc()); 
1667     COLORREF old_background 
= ::GetBkColor(GetHdc()); 
1668     if (m_textForegroundColour
.Ok()) 
1670         ::SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel() ); 
1672     if (m_textBackgroundColour
.Ok()) 
1674         ::SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() ); 
1677     DWORD dwRop 
= SRCCOPY
; 
1680         case wxXOR
:          dwRop 
= SRCINVERT
;        break; 
1681         case wxINVERT
:       dwRop 
= DSTINVERT
;        break; 
1682         case wxOR_REVERSE
:   dwRop 
= 0x00DD0228;       break; 
1683         case wxAND_REVERSE
:  dwRop 
= SRCERASE
;         break; 
1684         case wxCLEAR
:        dwRop 
= BLACKNESS
;        break; 
1685         case wxSET
:          dwRop 
= WHITENESS
;        break; 
1686         case wxOR_INVERT
:    dwRop 
= MERGEPAINT
;       break; 
1687         case wxAND
:          dwRop 
= SRCAND
;           break; 
1688         case wxOR
:           dwRop 
= SRCPAINT
;         break; 
1689         case wxEQUIV
:        dwRop 
= 0x00990066;       break; 
1690         case wxNAND
:         dwRop 
= 0x007700E6;       break; 
1691         case wxAND_INVERT
:   dwRop 
= 0x00220326;       break; 
1692         case wxCOPY
:         dwRop 
= SRCCOPY
;          break; 
1693         case wxNO_OP
:        dwRop 
= DSTCOPY
;          break; 
1694         case wxSRC_INVERT
:   dwRop 
= NOTSRCCOPY
;       break; 
1695         case wxNOR
:          dwRop 
= NOTSRCCOPY
;       break; 
1697            wxFAIL_MSG( wxT("unsupported logical function") ); 
1701     bool success 
= FALSE
; 
1706         // we want the part of the image corresponding to the mask to be 
1707         // transparent, so use "DSTCOPY" ROP for the mask points (the usual 
1708         // meaning of fg and bg is inverted which corresponds to wxWin notion 
1709         // of the mask which is also contrary to the Windows one) 
1711         // On some systems, MaskBlt succeeds yet is much much slower 
1712         // than the wxWindows fall-back implementation. So we need 
1713         // to be able to switch this on and off at runtime. 
1714 #if wxUSE_SYSTEM_OPTIONS 
1715         if (wxSystemOptions::GetOptionInt(wxT("no-maskblt")) == 0) 
1718            success 
= ::MaskBlt(GetHdc(), xdest
, ydest
, width
, height
, 
1719                             GetHdcOf(*source
), xsrc
, ysrc
, 
1720                             (HBITMAP
)mask
->GetMaskBitmap(), xsrcMask
, ysrcMask
, 
1721                             MAKEROP4(dwRop
, DSTCOPY
)) != 0; 
1727             // Blit bitmap with mask 
1730             HBITMAP buffer_bmap 
; 
1732 #if wxUSE_DC_CACHEING 
1733             // create a temp buffer bitmap and DCs to access it and the mask 
1734             wxDCCacheEntry
* dcCacheEntry1 
= FindDCInCache(NULL
, source
->GetHDC()); 
1735             dc_mask 
= (HDC
) dcCacheEntry1
->m_dc
; 
1737             wxDCCacheEntry
* dcCacheEntry2 
= FindDCInCache(dcCacheEntry1
, GetHDC()); 
1738             dc_buffer 
= (HDC
) dcCacheEntry2
->m_dc
; 
1740             wxDCCacheEntry
* bitmapCacheEntry 
= FindBitmapInCache(GetHDC(), 
1743             buffer_bmap 
= (HBITMAP
) bitmapCacheEntry
->m_bitmap
; 
1744 #else // !wxUSE_DC_CACHEING 
1745             // create a temp buffer bitmap and DCs to access it and the mask 
1746             dc_mask 
= ::CreateCompatibleDC(GetHdcOf(*source
)); 
1747             dc_buffer 
= ::CreateCompatibleDC(GetHdc()); 
1748             buffer_bmap 
= ::CreateCompatibleBitmap(GetHdc(), width
, height
); 
1749 #endif // wxUSE_DC_CACHEING/!wxUSE_DC_CACHEING 
1750             ::SelectObject(dc_mask
, (HBITMAP
) mask
->GetMaskBitmap()); 
1751             ::SelectObject(dc_buffer
, buffer_bmap
); 
1753             // copy dest to buffer 
1754             if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
, 
1755                            GetHdc(), xdest
, ydest
, SRCCOPY
) ) 
1757                 wxLogLastError(wxT("BitBlt")); 
1760             // copy src to buffer using selected raster op 
1761             if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
, 
1762                            GetHdcOf(*source
), xsrc
, ysrc
, dwRop
) ) 
1764                 wxLogLastError(wxT("BitBlt")); 
1767             // set masked area in buffer to BLACK (pixel value 0) 
1768             COLORREF prevBkCol 
= ::SetBkColor(GetHdc(), RGB(255, 255, 255)); 
1769             COLORREF prevCol 
= ::SetTextColor(GetHdc(), RGB(0, 0, 0)); 
1770             if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
, 
1771                            dc_mask
, xsrcMask
, ysrcMask
, SRCAND
) ) 
1773                 wxLogLastError(wxT("BitBlt")); 
1776             // set unmasked area in dest to BLACK 
1777             ::SetBkColor(GetHdc(), RGB(0, 0, 0)); 
1778             ::SetTextColor(GetHdc(), RGB(255, 255, 255)); 
1779             if ( !::BitBlt(GetHdc(), xdest
, ydest
, (int)width
, (int)height
, 
1780                            dc_mask
, xsrcMask
, ysrcMask
, SRCAND
) ) 
1782                 wxLogLastError(wxT("BitBlt")); 
1784             ::SetBkColor(GetHdc(), prevBkCol
);   // restore colours to original values 
1785             ::SetTextColor(GetHdc(), prevCol
); 
1787             // OR buffer to dest 
1788             success 
= ::BitBlt(GetHdc(), xdest
, ydest
, 
1789                                (int)width
, (int)height
, 
1790                                dc_buffer
, 0, 0, SRCPAINT
) != 0; 
1793                 wxLogLastError(wxT("BitBlt")); 
1796             // tidy up temporary DCs and bitmap 
1797             ::SelectObject(dc_mask
, 0); 
1798             ::SelectObject(dc_buffer
, 0); 
1800 #if !wxUSE_DC_CACHEING 
1802                 ::DeleteDC(dc_mask
); 
1803                 ::DeleteDC(dc_buffer
); 
1804                 ::DeleteObject(buffer_bmap
); 
1809     else // no mask, just BitBlt() it 
1811         success 
= ::BitBlt(GetHdc(), xdest
, ydest
, 
1812                            (int)width
, (int)height
, 
1813                            GetHdcOf(*source
), xsrc
, ysrc
, dwRop
) != 0; 
1816             wxLogLastError(wxT("BitBlt")); 
1819     ::SetTextColor(GetHdc(), old_textground
); 
1820     ::SetBkColor(GetHdc(), old_background
); 
1825 void wxDC::DoGetSize(int *w
, int *h
) const 
1827 #ifdef __WXMICROWIN__ 
1828     if (!GetHDC()) return; 
1831     if ( w 
) *w 
= ::GetDeviceCaps(GetHdc(), HORZRES
); 
1832     if ( h 
) *h 
= ::GetDeviceCaps(GetHdc(), VERTRES
); 
1835 void wxDC::DoGetSizeMM(int *w
, int *h
) const 
1837 #ifdef __WXMICROWIN__ 
1838     if (!GetHDC()) return; 
1841     if ( w 
) *w 
= ::GetDeviceCaps(GetHdc(), HORZSIZE
); 
1842     if ( h 
) *h 
= ::GetDeviceCaps(GetHdc(), VERTSIZE
); 
1845 wxSize 
wxDC::GetPPI() const 
1847 #ifdef __WXMICROWIN__ 
1848     if (!GetHDC()) return wxSize(); 
1851     int x 
= ::GetDeviceCaps(GetHdc(), LOGPIXELSX
); 
1852     int y 
= ::GetDeviceCaps(GetHdc(), LOGPIXELSY
); 
1854     return wxSize(x
, y
); 
1857 // For use by wxWindows only, unless custom units are required. 
1858 void wxDC::SetLogicalScale(double x
, double y
) 
1860 #ifdef __WXMICROWIN__ 
1861     if (!GetHDC()) return; 
1864     m_logicalScaleX 
= x
; 
1865     m_logicalScaleY 
= y
; 
1868 #if WXWIN_COMPATIBILITY 
1869 void wxDC::DoGetTextExtent(const wxString
& string
, float *x
, float *y
, 
1870                          float *descent
, float *externalLeading
, 
1871                          wxFont 
*theFont
, bool use16bit
) const 
1873 #ifdef __WXMICROWIN__ 
1874     if (!GetHDC()) return; 
1877     wxCoord x1
, y1
, descent1
, externalLeading1
; 
1878     GetTextExtent(string
, & x1
, & y1
, & descent1
, & externalLeading1
, theFont
, use16bit
); 
1881         *descent 
= descent1
; 
1882     if (externalLeading
) 
1883         *externalLeading 
= externalLeading1
; 
1887 #if wxUSE_DC_CACHEING 
1890  * This implementation is a bit ugly and uses the old-fashioned wxList class, so I will 
1891  * improve it in due course, either using arrays, or simply storing pointers to one 
1892  * entry for the bitmap, and two for the DCs. -- JACS 
1895 wxList 
wxDC::sm_bitmapCache
; 
1896 wxList 
wxDC::sm_dcCache
; 
1898 wxDCCacheEntry::wxDCCacheEntry(WXHBITMAP hBitmap
, int w
, int h
, int depth
) 
1907 wxDCCacheEntry::wxDCCacheEntry(WXHDC hDC
, int depth
) 
1916 wxDCCacheEntry::~wxDCCacheEntry() 
1919         ::DeleteObject((HBITMAP
) m_bitmap
); 
1921         ::DeleteDC((HDC
) m_dc
); 
1924 wxDCCacheEntry
* wxDC::FindBitmapInCache(WXHDC dc
, int w
, int h
) 
1926     int depth 
= ::GetDeviceCaps((HDC
) dc
, PLANES
) * ::GetDeviceCaps((HDC
) dc
, BITSPIXEL
); 
1927     wxNode
* node 
= sm_bitmapCache
.First(); 
1930         wxDCCacheEntry
* entry 
= (wxDCCacheEntry
*) node
->Data(); 
1932         if (entry
->m_depth 
== depth
) 
1934             if (entry
->m_width 
< w 
|| entry
->m_height 
< h
) 
1936                 ::DeleteObject((HBITMAP
) entry
->m_bitmap
); 
1937                 entry
->m_bitmap 
= (WXHBITMAP
) ::CreateCompatibleBitmap((HDC
) dc
, w
, h
); 
1938                 if ( !entry
->m_bitmap
) 
1940                     wxLogLastError(wxT("CreateCompatibleBitmap")); 
1942                 entry
->m_width 
= w
; entry
->m_height 
= h
; 
1948         node 
= node
->Next(); 
1950     WXHBITMAP hBitmap 
= (WXHBITMAP
) ::CreateCompatibleBitmap((HDC
) dc
, w
, h
); 
1953         wxLogLastError(wxT("CreateCompatibleBitmap")); 
1955     wxDCCacheEntry
* entry 
= new wxDCCacheEntry(hBitmap
, w
, h
, depth
); 
1956     AddToBitmapCache(entry
); 
1960 wxDCCacheEntry
* wxDC::FindDCInCache(wxDCCacheEntry
* notThis
, WXHDC dc
) 
1962     int depth 
= ::GetDeviceCaps((HDC
) dc
, PLANES
) * ::GetDeviceCaps((HDC
) dc
, BITSPIXEL
); 
1963     wxNode
* node 
= sm_dcCache
.First(); 
1966         wxDCCacheEntry
* entry 
= (wxDCCacheEntry
*) node
->Data(); 
1968         // Don't return the same one as we already have 
1969         if (!notThis 
|| (notThis 
!= entry
)) 
1971             if (entry
->m_depth 
== depth
) 
1977         node 
= node
->Next(); 
1979     WXHDC hDC 
= (WXHDC
) ::CreateCompatibleDC((HDC
) dc
); 
1982         wxLogLastError(wxT("CreateCompatibleDC")); 
1984     wxDCCacheEntry
* entry 
= new wxDCCacheEntry(hDC
, depth
); 
1985     AddToDCCache(entry
); 
1989 void wxDC::AddToBitmapCache(wxDCCacheEntry
* entry
) 
1991     sm_bitmapCache
.Append(entry
); 
1994 void wxDC::AddToDCCache(wxDCCacheEntry
* entry
) 
1996     sm_dcCache
.Append(entry
); 
1999 void wxDC::ClearCache() 
2001     sm_bitmapCache
.DeleteContents(TRUE
); 
2002     sm_bitmapCache
.Clear(); 
2003     sm_bitmapCache
.DeleteContents(FALSE
); 
2004     sm_dcCache
.DeleteContents(TRUE
); 
2006     sm_dcCache
.DeleteContents(FALSE
); 
2009 // Clean up cache at app exit 
2010 class wxDCModule 
: public wxModule
 
2013     virtual bool OnInit() { return TRUE
; } 
2014     virtual void OnExit() { wxDC::ClearCache(); } 
2017     DECLARE_DYNAMIC_CLASS(wxDCModule
) 
2020 IMPLEMENT_DYNAMIC_CLASS(wxDCModule
, wxModule
) 
2023     // wxUSE_DC_CACHEING