1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
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"
46 #include "wx/dynload.h"
47 #include "wx/rawbmp.h"
52 #include "wx/msw/private.h" // needs to be before #include <commdlg.h>
54 #if wxUSE_COMMON_DIALOGS && !defined(__WXMICROWIN__)
63 #define AC_SRC_ALPHA 1
66 /* Quaternary raster codes */
68 #define MAKEROP4(fore,back) (DWORD)((((back) << 8) & 0xFF000000) | (fore))
71 IMPLEMENT_ABSTRACT_CLASS(wxDC
, wxDCBase
)
73 // ---------------------------------------------------------------------------
75 // ---------------------------------------------------------------------------
77 static const int VIEWPORT_EXTENT
= 1000;
79 static const int MM_POINTS
= 9;
80 static const int MM_METRIC
= 10;
82 // usually this is defined in math.h
84 static const double M_PI
= 3.14159265358979323846;
87 // ROPs which don't have standard names (see "Ternary Raster Operations" in the
88 // MSDN docs for how this and other numbers in wxDC::Blit() are obtained)
89 #define DSTCOPY 0x00AA0029 // a.k.a. NOP operation
91 // ----------------------------------------------------------------------------
92 // macros for logical <-> device coords conversion
93 // ----------------------------------------------------------------------------
96 We currently let Windows do all the translations itself so these macros are
97 not really needed (any more) but keep them to enhance readability of the
98 code by allowing to see where are the logical and where are the device
103 #define XLOG2DEV(x) (x)
104 #define YLOG2DEV(y) (y)
107 #define XDEV2LOG(x) (x)
108 #define YDEV2LOG(y) (y)
110 // ---------------------------------------------------------------------------
112 // ---------------------------------------------------------------------------
114 // convert degrees to radians
115 static inline double DegToRad(double deg
) { return (deg
* M_PI
) / 180.0; }
117 // our (limited) AlphaBlend() replacement
119 wxAlphaBlend(wxDC
& dc
, int x
, int y
, int w
, int h
, const wxBitmap
& bmp
);
121 // ----------------------------------------------------------------------------
123 // ----------------------------------------------------------------------------
125 // instead of duplicating the same code which sets and then restores text
126 // colours in each wxDC method working with wxSTIPPLE_MASK_OPAQUE brushes,
127 // encapsulate this in a small helper class
129 // wxColourChanger: changes the text colours in the ctor if required and
130 // restores them in the dtor
131 class wxColourChanger
134 wxColourChanger(wxDC
& dc
);
140 COLORREF m_colFgOld
, m_colBgOld
;
145 // this class saves the old stretch blit mode during its life time
146 class StretchBltModeChanger
149 StretchBltModeChanger(HDC hdc
, int mode
)
152 m_modeOld
= ::SetStretchBltMode(m_hdc
, mode
);
154 wxLogLastError(_T("SetStretchBltMode"));
157 ~StretchBltModeChanger()
159 if ( !::SetStretchBltMode(m_hdc
, m_modeOld
) )
160 wxLogLastError(_T("SetStretchBltMode"));
169 // ===========================================================================
171 // ===========================================================================
173 // ----------------------------------------------------------------------------
175 // ----------------------------------------------------------------------------
177 wxColourChanger::wxColourChanger(wxDC
& dc
) : m_dc(dc
)
179 const wxBrush
& brush
= dc
.GetBrush();
180 if ( brush
.Ok() && brush
.GetStyle() == wxSTIPPLE_MASK_OPAQUE
)
182 HDC hdc
= GetHdcOf(dc
);
183 m_colFgOld
= ::GetTextColor(hdc
);
184 m_colBgOld
= ::GetBkColor(hdc
);
186 // note that Windows convention is opposite to wxWindows one, this is
187 // why text colour becomes the background one and vice versa
188 const wxColour
& colFg
= dc
.GetTextForeground();
191 ::SetBkColor(hdc
, colFg
.GetPixel());
194 const wxColour
& colBg
= dc
.GetTextBackground();
197 ::SetTextColor(hdc
, colBg
.GetPixel());
201 dc
.GetBackgroundMode() == wxTRANSPARENT
? TRANSPARENT
204 // flag which telsl us to undo changes in the dtor
209 // nothing done, nothing to undo
214 wxColourChanger::~wxColourChanger()
218 // restore the colours we changed
219 HDC hdc
= GetHdcOf(m_dc
);
221 ::SetBkMode(hdc
, TRANSPARENT
);
222 ::SetTextColor(hdc
, m_colFgOld
);
223 ::SetBkColor(hdc
, m_colBgOld
);
227 // ---------------------------------------------------------------------------
229 // ---------------------------------------------------------------------------
231 // Default constructor
242 #endif // wxUSE_PALETTE
252 SelectOldObjects(m_hDC
);
254 // if we own the HDC, we delete it, otherwise we just release it
258 ::DeleteDC(GetHdc());
260 else // we don't own our HDC
264 ::ReleaseDC(GetHwndOf(m_canvas
), GetHdc());
268 // Must have been a wxScreenDC
269 ::ReleaseDC((HWND
) NULL
, GetHdc());
275 // This will select current objects out of the DC,
276 // which is what you have to do before deleting the
278 void wxDC::SelectOldObjects(WXHDC dc
)
284 ::SelectObject((HDC
) dc
, (HBITMAP
) m_oldBitmap
);
286 if (m_selectedBitmap
.Ok())
288 m_selectedBitmap
.SetSelectedInto(NULL
);
295 ::SelectObject((HDC
) dc
, (HPEN
) m_oldPen
);
300 ::SelectObject((HDC
) dc
, (HBRUSH
) m_oldBrush
);
305 ::SelectObject((HDC
) dc
, (HFONT
) m_oldFont
);
312 ::SelectPalette((HDC
) dc
, (HPALETTE
) m_oldPalette
, FALSE
);
315 #endif // wxUSE_PALETTE
318 m_brush
= wxNullBrush
;
321 m_palette
= wxNullPalette
;
322 #endif // wxUSE_PALETTE
324 m_backgroundBrush
= wxNullBrush
;
325 m_selectedBitmap
= wxNullBitmap
;
328 // ---------------------------------------------------------------------------
330 // ---------------------------------------------------------------------------
332 void wxDC::UpdateClipBox()
334 #ifdef __WXMICROWIN__
335 if (!GetHDC()) return;
339 ::GetClipBox(GetHdc(), &rect
);
341 m_clipX1
= (wxCoord
) XDEV2LOG(rect
.left
);
342 m_clipY1
= (wxCoord
) YDEV2LOG(rect
.top
);
343 m_clipX2
= (wxCoord
) XDEV2LOG(rect
.right
);
344 m_clipY2
= (wxCoord
) YDEV2LOG(rect
.bottom
);
347 // common part of DoSetClippingRegion() and DoSetClippingRegionAsRegion()
348 void wxDC::SetClippingHrgn(WXHRGN hrgn
)
350 wxCHECK_RET( hrgn
, wxT("invalid clipping region") );
352 #ifdef __WXMICROWIN__
353 if (!GetHdc()) return;
354 #endif // __WXMICROWIN__
356 // note that we combine the new clipping region with the existing one: this
357 // is compatible with what the other ports do and is the documented
358 // behaviour now (starting with 2.3.3)
361 if ( !::GetClipBox(GetHdc(), &rectClip
) )
364 HRGN hrgnDest
= ::CreateRectRgn(0, 0, 0, 0);
365 HRGN hrgnClipOld
= ::CreateRectRgn(rectClip
.left
, rectClip
.top
,
366 rectClip
.right
, rectClip
.bottom
);
368 if ( ::CombineRgn(hrgnDest
, hrgnClipOld
, (HRGN
)hrgn
, RGN_AND
) != ERROR
)
370 ::SelectClipRgn(GetHdc(), hrgnDest
);
373 ::DeleteObject(hrgnClipOld
);
374 ::DeleteObject(hrgnDest
);
376 if ( ::ExtSelectClipRgn(GetHdc(), (HRGN
)hrgn
, RGN_AND
) == ERROR
)
378 wxLogLastError(_T("ExtSelectClipRgn"));
389 void wxDC::DoSetClippingRegion(wxCoord x
, wxCoord y
, wxCoord w
, wxCoord h
)
391 // the region coords are always the device ones, so do the translation
394 // FIXME: possible +/-1 error here, to check!
395 HRGN hrgn
= ::CreateRectRgn(LogicalToDeviceX(x
),
397 LogicalToDeviceX(x
+ w
),
398 LogicalToDeviceY(y
+ h
));
401 wxLogLastError(_T("CreateRectRgn"));
405 SetClippingHrgn((WXHRGN
)hrgn
);
407 ::DeleteObject(hrgn
);
411 void wxDC::DoSetClippingRegionAsRegion(const wxRegion
& region
)
413 SetClippingHrgn(region
.GetHRGN());
416 void wxDC::DestroyClippingRegion()
418 #ifdef __WXMICROWIN__
419 if (!GetHDC()) return;
422 if (m_clipping
&& m_hDC
)
424 // TODO: this should restore the previous clipping region,
425 // so that OnPaint processing works correctly, and the update
426 // clipping region doesn't get destroyed after the first
427 // DestroyClippingRegion.
428 HRGN rgn
= CreateRectRgn(0, 0, 32000, 32000);
429 ::SelectClipRgn(GetHdc(), rgn
);
436 // ---------------------------------------------------------------------------
437 // query capabilities
438 // ---------------------------------------------------------------------------
440 bool wxDC::CanDrawBitmap() const
445 bool wxDC::CanGetTextExtent() const
447 #ifdef __WXMICROWIN__
448 // TODO Extend MicroWindows' GetDeviceCaps function
451 // What sort of display is it?
452 int technology
= ::GetDeviceCaps(GetHdc(), TECHNOLOGY
);
454 return (technology
== DT_RASDISPLAY
) || (technology
== DT_RASPRINTER
);
458 int wxDC::GetDepth() const
460 #ifdef __WXMICROWIN__
461 if (!GetHDC()) return 16;
464 return (int)::GetDeviceCaps(GetHdc(), BITSPIXEL
);
467 // ---------------------------------------------------------------------------
469 // ---------------------------------------------------------------------------
473 #ifdef __WXMICROWIN__
474 if (!GetHDC()) return;
480 GetClientRect((HWND
) m_canvas
->GetHWND(), &rect
);
484 // No, I think we should simply ignore this if printing on e.g.
486 // wxCHECK_RET( m_selectedBitmap.Ok(), wxT("this DC can't be cleared") );
487 if (!m_selectedBitmap
.Ok())
490 rect
.left
= 0; rect
.top
= 0;
491 rect
.right
= m_selectedBitmap
.GetWidth();
492 rect
.bottom
= m_selectedBitmap
.GetHeight();
495 (void) ::SetMapMode(GetHdc(), MM_TEXT
);
497 DWORD colour
= ::GetBkColor(GetHdc());
498 HBRUSH brush
= ::CreateSolidBrush(colour
);
499 ::FillRect(GetHdc(), &rect
, brush
);
500 ::DeleteObject(brush
);
502 int width
= DeviceToLogicalXRel(VIEWPORT_EXTENT
)*m_signX
,
503 height
= DeviceToLogicalYRel(VIEWPORT_EXTENT
)*m_signY
;
505 ::SetMapMode(GetHdc(), MM_ANISOTROPIC
);
506 ::SetViewportExtEx(GetHdc(), VIEWPORT_EXTENT
, VIEWPORT_EXTENT
, NULL
);
507 ::SetWindowExtEx(GetHdc(), width
, height
, NULL
);
508 ::SetViewportOrgEx(GetHdc(), (int)m_deviceOriginX
, (int)m_deviceOriginY
, NULL
);
509 ::SetWindowOrgEx(GetHdc(), (int)m_logicalOriginX
, (int)m_logicalOriginY
, NULL
);
512 bool wxDC::DoFloodFill(wxCoord x
, wxCoord y
, const wxColour
& col
, int style
)
514 #ifdef __WXMICROWIN__
515 if (!GetHDC()) return FALSE
;
518 bool success
= (0 != ::ExtFloodFill(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
),
520 style
== wxFLOOD_SURFACE
? FLOODFILLSURFACE
521 : FLOODFILLBORDER
) ) ;
524 // quoting from the MSDN docs:
526 // Following are some of the reasons this function might fail:
528 // * The filling could not be completed.
529 // * The specified point has the boundary color specified by the
530 // crColor parameter (if FLOODFILLBORDER was requested).
531 // * The specified point does not have the color specified by
532 // crColor (if FLOODFILLSURFACE was requested)
533 // * The point is outside the clipping region that is, it is not
534 // visible on the device.
536 wxLogLastError(wxT("ExtFloodFill"));
539 CalcBoundingBox(x
, y
);
544 bool wxDC::DoGetPixel(wxCoord x
, wxCoord y
, wxColour
*col
) const
546 #ifdef __WXMICROWIN__
547 if (!GetHDC()) return FALSE
;
550 wxCHECK_MSG( col
, FALSE
, _T("NULL colour parameter in wxDC::GetPixel") );
552 // get the color of the pixel
553 COLORREF pixelcolor
= ::GetPixel(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
));
555 wxRGBToColour(*col
, pixelcolor
);
560 void wxDC::DoCrossHair(wxCoord x
, wxCoord y
)
562 #ifdef __WXMICROWIN__
563 if (!GetHDC()) return;
566 wxCoord x1
= x
-VIEWPORT_EXTENT
;
567 wxCoord y1
= y
-VIEWPORT_EXTENT
;
568 wxCoord x2
= x
+VIEWPORT_EXTENT
;
569 wxCoord y2
= y
+VIEWPORT_EXTENT
;
571 (void)MoveToEx(GetHdc(), XLOG2DEV(x1
), YLOG2DEV(y
), NULL
);
572 (void)LineTo(GetHdc(), XLOG2DEV(x2
), YLOG2DEV(y
));
574 (void)MoveToEx(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y1
), NULL
);
575 (void)LineTo(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y2
));
577 CalcBoundingBox(x1
, y1
);
578 CalcBoundingBox(x2
, y2
);
581 void wxDC::DoDrawLine(wxCoord x1
, wxCoord y1
, wxCoord x2
, wxCoord y2
)
583 #ifdef __WXMICROWIN__
584 if (!GetHDC()) return;
587 (void)MoveToEx(GetHdc(), XLOG2DEV(x1
), YLOG2DEV(y1
), NULL
);
588 (void)LineTo(GetHdc(), XLOG2DEV(x2
), YLOG2DEV(y2
));
590 CalcBoundingBox(x1
, y1
);
591 CalcBoundingBox(x2
, y2
);
594 // Draws an arc of a circle, centred on (xc, yc), with starting point (x1, y1)
595 // and ending at (x2, y2)
596 void wxDC::DoDrawArc(wxCoord x1
, wxCoord y1
,
597 wxCoord x2
, wxCoord y2
,
598 wxCoord xc
, wxCoord yc
)
600 #ifdef __WXMICROWIN__
601 if (!GetHDC()) return;
604 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
608 double radius
= (double)sqrt(dx
*dx
+dy
*dy
);
609 wxCoord r
= (wxCoord
)radius
;
611 // treat the special case of full circle separately
612 if ( x1
== x2
&& y1
== y2
)
614 DrawEllipse(xc
- r
, yc
- r
, 2*r
, 2*r
);
618 wxCoord xx1
= XLOG2DEV(x1
);
619 wxCoord yy1
= YLOG2DEV(y1
);
620 wxCoord xx2
= XLOG2DEV(x2
);
621 wxCoord yy2
= YLOG2DEV(y2
);
622 wxCoord xxc
= XLOG2DEV(xc
);
623 wxCoord yyc
= YLOG2DEV(yc
);
624 wxCoord ray
= (wxCoord
) sqrt(double((xxc
-xx1
)*(xxc
-xx1
)+(yyc
-yy1
)*(yyc
-yy1
)));
626 wxCoord xxx1
= (wxCoord
) (xxc
-ray
);
627 wxCoord yyy1
= (wxCoord
) (yyc
-ray
);
628 wxCoord xxx2
= (wxCoord
) (xxc
+ray
);
629 wxCoord yyy2
= (wxCoord
) (yyc
+ray
);
631 if ( m_brush
.Ok() && m_brush
.GetStyle() != wxTRANSPARENT
)
633 // Have to add 1 to bottom-right corner of rectangle
634 // to make semi-circles look right (crooked line otherwise).
635 // Unfortunately this is not a reliable method, depends
636 // on the size of shape.
637 // TODO: figure out why this happens!
638 Pie(GetHdc(),xxx1
,yyy1
,xxx2
+1,yyy2
+1, xx1
,yy1
,xx2
,yy2
);
642 Arc(GetHdc(),xxx1
,yyy1
,xxx2
,yyy2
, xx1
,yy1
,xx2
,yy2
);
645 CalcBoundingBox(xc
- r
, yc
- r
);
646 CalcBoundingBox(xc
+ r
, yc
+ r
);
649 void wxDC::DoDrawCheckMark(wxCoord x1
, wxCoord y1
,
650 wxCoord width
, wxCoord height
)
652 #ifdef __WXMICROWIN__
653 if (!GetHDC()) return;
656 wxCoord x2
= x1
+ width
,
659 #if defined(__WIN32__) && !defined(__SC__) && !defined(__WXMICROWIN__)
666 DrawFrameControl(GetHdc(), &rect
, DFC_MENU
, DFCS_MENUCHECK
);
668 // In WIN16, draw a cross
669 HPEN blackPen
= ::CreatePen(PS_SOLID
, 1, RGB(0, 0, 0));
670 HPEN whiteBrush
= (HPEN
)::GetStockObject(WHITE_BRUSH
);
671 HPEN hPenOld
= (HPEN
)::SelectObject(GetHdc(), blackPen
);
672 HPEN hBrushOld
= (HPEN
)::SelectObject(GetHdc(), whiteBrush
);
673 ::SetROP2(GetHdc(), R2_COPYPEN
);
674 Rectangle(GetHdc(), x1
, y1
, x2
, y2
);
675 MoveToEx(GetHdc(), x1
, y1
, NULL
);
676 LineTo(GetHdc(), x2
, y2
);
677 MoveToEx(GetHdc(), x2
, y1
, NULL
);
678 LineTo(GetHdc(), x1
, y2
);
679 ::SelectObject(GetHdc(), hPenOld
);
680 ::SelectObject(GetHdc(), hBrushOld
);
681 ::DeleteObject(blackPen
);
684 CalcBoundingBox(x1
, y1
);
685 CalcBoundingBox(x2
, y2
);
688 void wxDC::DoDrawPoint(wxCoord x
, wxCoord y
)
690 #ifdef __WXMICROWIN__
691 if (!GetHDC()) return;
694 COLORREF color
= 0x00ffffff;
697 color
= m_pen
.GetColour().GetPixel();
700 SetPixel(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), color
);
702 CalcBoundingBox(x
, y
);
705 void wxDC::DoDrawPolygon(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
,int fillStyle
)
707 #ifdef __WXMICROWIN__
708 if (!GetHDC()) return;
711 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
713 // Do things less efficiently if we have offsets
714 if (xoffset
!= 0 || yoffset
!= 0)
716 POINT
*cpoints
= new POINT
[n
];
718 for (i
= 0; i
< n
; i
++)
720 cpoints
[i
].x
= (int)(points
[i
].x
+ xoffset
);
721 cpoints
[i
].y
= (int)(points
[i
].y
+ yoffset
);
723 CalcBoundingBox(cpoints
[i
].x
, cpoints
[i
].y
);
725 int prev
= SetPolyFillMode(GetHdc(),fillStyle
==wxODDEVEN_RULE
?ALTERNATE
:WINDING
);
726 (void)Polygon(GetHdc(), cpoints
, n
);
727 SetPolyFillMode(GetHdc(),prev
);
733 for (i
= 0; i
< n
; i
++)
734 CalcBoundingBox(points
[i
].x
, points
[i
].y
);
736 int prev
= SetPolyFillMode(GetHdc(),fillStyle
==wxODDEVEN_RULE
?ALTERNATE
:WINDING
);
737 (void)Polygon(GetHdc(), (POINT
*) points
, n
);
738 SetPolyFillMode(GetHdc(),prev
);
742 void wxDC::DoDrawLines(int n
, wxPoint points
[], wxCoord xoffset
, wxCoord yoffset
)
744 #ifdef __WXMICROWIN__
745 if (!GetHDC()) return;
748 // Do things less efficiently if we have offsets
749 if (xoffset
!= 0 || yoffset
!= 0)
751 POINT
*cpoints
= new POINT
[n
];
753 for (i
= 0; i
< n
; i
++)
755 cpoints
[i
].x
= (int)(points
[i
].x
+ xoffset
);
756 cpoints
[i
].y
= (int)(points
[i
].y
+ yoffset
);
758 CalcBoundingBox(cpoints
[i
].x
, cpoints
[i
].y
);
760 (void)Polyline(GetHdc(), cpoints
, n
);
766 for (i
= 0; i
< n
; i
++)
767 CalcBoundingBox(points
[i
].x
, points
[i
].y
);
769 (void)Polyline(GetHdc(), (POINT
*) points
, n
);
773 void wxDC::DoDrawRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
)
775 #ifdef __WXMICROWIN__
776 if (!GetHDC()) return;
779 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
781 wxCoord x2
= x
+ width
;
782 wxCoord y2
= y
+ height
;
784 if ((m_logicalFunction
== wxCOPY
) && (m_pen
.GetStyle() == wxTRANSPARENT
))
787 rect
.left
= XLOG2DEV(x
);
788 rect
.top
= YLOG2DEV(y
);
789 rect
.right
= XLOG2DEV(x2
);
790 rect
.bottom
= YLOG2DEV(y2
);
791 (void)FillRect(GetHdc(), &rect
, (HBRUSH
)m_brush
.GetResourceHandle() );
795 // Windows draws the filled rectangles without outline (i.e. drawn with a
796 // transparent pen) one pixel smaller in both directions and we want them
797 // to have the same size regardless of which pen is used - adjust
799 // I wonder if this shouldn´t be done after the LOG2DEV() conversions. RR.
800 if ( m_pen
.GetStyle() == wxTRANSPARENT
)
806 (void)Rectangle(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
));
810 CalcBoundingBox(x
, y
);
811 CalcBoundingBox(x2
, y2
);
814 void wxDC::DoDrawRoundedRectangle(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
, double radius
)
816 #ifdef __WXMICROWIN__
817 if (!GetHDC()) return;
820 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
822 // Now, a negative radius value is interpreted to mean
823 // 'the proportion of the smallest X or Y dimension'
827 double smallest
= 0.0;
832 radius
= (- radius
* smallest
);
835 wxCoord x2
= (x
+width
);
836 wxCoord y2
= (y
+height
);
838 // Windows draws the filled rectangles without outline (i.e. drawn with a
839 // transparent pen) one pixel smaller in both directions and we want them
840 // to have the same size regardless of which pen is used - adjust
841 if ( m_pen
.GetStyle() == wxTRANSPARENT
)
847 (void)RoundRect(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
),
848 YLOG2DEV(y2
), (int) (2*XLOG2DEV(radius
)), (int)( 2*YLOG2DEV(radius
)));
850 CalcBoundingBox(x
, y
);
851 CalcBoundingBox(x2
, y2
);
854 void wxDC::DoDrawEllipse(wxCoord x
, wxCoord y
, wxCoord width
, wxCoord height
)
856 #ifdef __WXMICROWIN__
857 if (!GetHDC()) return;
860 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
862 wxCoord x2
= (x
+width
);
863 wxCoord y2
= (y
+height
);
865 (void)Ellipse(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
));
867 CalcBoundingBox(x
, y
);
868 CalcBoundingBox(x2
, y2
);
871 // Chris Breeze 20/5/98: first implementation of DrawEllipticArc on Windows
872 void wxDC::DoDrawEllipticArc(wxCoord x
,wxCoord y
,wxCoord w
,wxCoord h
,double sa
,double ea
)
874 #ifdef __WXMICROWIN__
875 if (!GetHDC()) return;
878 wxColourChanger
cc(*this); // needed for wxSTIPPLE_MASK_OPAQUE handling
883 int rx1
= XLOG2DEV(x
+w
/2);
884 int ry1
= YLOG2DEV(y
+h
/2);
891 rx1
+= (int)(100.0 * abs(w
) * cos(sa
));
892 ry1
-= (int)(100.0 * abs(h
) * m_signY
* sin(sa
));
893 rx2
+= (int)(100.0 * abs(w
) * cos(ea
));
894 ry2
-= (int)(100.0 * abs(h
) * m_signY
* sin(ea
));
896 // draw pie with NULL_PEN first and then outline otherwise a line is
897 // drawn from the start and end points to the centre
898 HPEN hpenOld
= (HPEN
) ::SelectObject(GetHdc(), (HPEN
) ::GetStockObject(NULL_PEN
));
901 (void)Pie(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
)+1, YLOG2DEV(y2
)+1,
906 (void)Pie(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
)-1, XLOG2DEV(x2
)+1, YLOG2DEV(y2
),
907 rx1
, ry1
-1, rx2
, ry2
-1);
910 ::SelectObject(GetHdc(), hpenOld
);
912 (void)Arc(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), XLOG2DEV(x2
), YLOG2DEV(y2
),
915 CalcBoundingBox(x
, y
);
916 CalcBoundingBox(x2
, y2
);
919 void wxDC::DoDrawIcon(const wxIcon
& icon
, wxCoord x
, wxCoord y
)
921 #ifdef __WXMICROWIN__
922 if (!GetHDC()) return;
925 wxCHECK_RET( icon
.Ok(), wxT("invalid icon in DrawIcon") );
928 ::DrawIconEx(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), GetHiconOf(icon
), icon
.GetWidth(), icon
.GetHeight(), 0, NULL
, DI_NORMAL
);
930 ::DrawIcon(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
), GetHiconOf(icon
));
933 CalcBoundingBox(x
, y
);
934 CalcBoundingBox(x
+ icon
.GetWidth(), y
+ icon
.GetHeight());
937 void wxDC::DoDrawBitmap( const wxBitmap
&bmp
, wxCoord x
, wxCoord y
, bool useMask
)
939 #ifdef __WXMICROWIN__
940 if (!GetHDC()) return;
943 wxCHECK_RET( bmp
.Ok(), _T("invalid bitmap in wxDC::DrawBitmap") );
945 int width
= bmp
.GetWidth(),
946 height
= bmp
.GetHeight();
948 HBITMAP hbmpMask
= 0;
952 #endif // wxUSE_PALETTE
954 // do we have AlphaBlend() and company in the headers?
956 if ( bmp
.HasAlpha() )
958 // yes, now try to see if we have it during run-time
960 typedef BOOL (WINAPI
*AlphaBlend_t
)(HDC
,int,int,int,int,
964 // bitmaps can be drawn only from GUI thread so there is no need to
965 // protect this static variable from multiple threads
966 static bool s_triedToLoad
= FALSE
;
967 static AlphaBlend_t pfnAlphaBlend
= NULL
;
968 if ( !s_triedToLoad
)
970 s_triedToLoad
= TRUE
;
972 // don't give errors about the DLL being unavailable, we're
973 // prepared to handle this
976 wxDynamicLibrary
dll(_T("msimg32.dll"));
977 if ( dll
.IsLoaded() )
979 pfnAlphaBlend
= (AlphaBlend_t
)dll
.GetSymbol(_T("AlphaBlend"));
982 // we must keep the DLL loaded if we want to be able to
983 // call AlphaBlend() so just never unload it at all, not a
993 SelectInHDC
select(hdcMem
, GetHbitmapOf(bmp
));
996 bf
.BlendOp
= AC_SRC_OVER
;
998 bf
.SourceConstantAlpha
= 0xff;
999 bf
.AlphaFormat
= AC_SRC_ALPHA
;
1001 if ( !pfnAlphaBlend(GetHdc(), x
, y
, width
, height
,
1002 hdcMem
, 0, 0, width
, height
,
1005 wxLogLastError(_T("AlphaBlend"));
1008 else // use our own (probably much slower) implementation
1010 wxAlphaBlend(*this, x
, y
, width
, height
, bmp
);
1015 #endif // defined(AC_SRC_OVER)
1019 wxMask
*mask
= bmp
.GetMask();
1021 hbmpMask
= (HBITMAP
)mask
->GetMaskBitmap();
1025 // don't give assert here because this would break existing
1026 // programs - just silently ignore useMask parameter
1033 // use MaskBlt() with ROP which doesn't do anything to dst in the mask
1035 // On some systems, MaskBlt succeeds yet is much much slower
1036 // than the wxWindows fall-back implementation. So we need
1037 // to be able to switch this on and off at runtime.
1039 #if wxUSE_SYSTEM_OPTIONS
1040 if (wxSystemOptions::GetOptionInt(wxT("no-maskblt")) == 0)
1044 HDC hdcMem
= ::CreateCompatibleDC(GetHdc());
1045 HGDIOBJ hOldBitmap
= ::SelectObject(hdcMem
, GetHbitmapOf(bmp
));
1047 wxPalette
*pal
= bmp
.GetPalette();
1048 if ( pal
&& ::GetDeviceCaps(cdc
,BITSPIXEL
) <= 8 )
1050 oldPal
= ::SelectPalette(hdcMem
, GetHpaletteOf(*pal
), FALSE
);
1051 ::RealizePalette(hdcMem
);
1053 #endif // wxUSE_PALETTE
1055 ok
= ::MaskBlt(cdc
, x
, y
, width
, height
,
1058 MAKEROP4(SRCCOPY
, DSTCOPY
)) != 0;
1062 ::SelectPalette(hdcMem
, oldPal
, FALSE
);
1063 #endif // wxUSE_PALETTE
1065 ::SelectObject(hdcMem
, hOldBitmap
);
1072 // Rather than reproduce wxDC::Blit, let's do it at the wxWin API
1075 memDC
.SelectObject(bmp
);
1077 Blit(x
, y
, width
, height
, &memDC
, 0, 0, wxCOPY
, useMask
);
1079 memDC
.SelectObject(wxNullBitmap
);
1082 else // no mask, just use BitBlt()
1085 HDC memdc
= ::CreateCompatibleDC( cdc
);
1086 HBITMAP hbitmap
= (HBITMAP
) bmp
.GetHBITMAP( );
1088 wxASSERT_MSG( hbitmap
, wxT("bitmap is ok but HBITMAP is NULL?") );
1090 COLORREF old_textground
= ::GetTextColor(GetHdc());
1091 COLORREF old_background
= ::GetBkColor(GetHdc());
1092 if (m_textForegroundColour
.Ok())
1094 ::SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel() );
1096 if (m_textBackgroundColour
.Ok())
1098 ::SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() );
1102 wxPalette
*pal
= bmp
.GetPalette();
1103 if ( pal
&& ::GetDeviceCaps(cdc
,BITSPIXEL
) <= 8 )
1105 oldPal
= ::SelectPalette(memdc
, GetHpaletteOf(*pal
), FALSE
);
1106 ::RealizePalette(memdc
);
1108 #endif // wxUSE_PALETTE
1110 HGDIOBJ hOldBitmap
= ::SelectObject( memdc
, hbitmap
);
1111 ::BitBlt( cdc
, x
, y
, width
, height
, memdc
, 0, 0, SRCCOPY
);
1115 ::SelectPalette(memdc
, oldPal
, FALSE
);
1116 #endif // wxUSE_PALETTE
1118 ::SelectObject( memdc
, hOldBitmap
);
1119 ::DeleteDC( memdc
);
1121 ::SetTextColor(GetHdc(), old_textground
);
1122 ::SetBkColor(GetHdc(), old_background
);
1126 void wxDC::DoDrawText(const wxString
& text
, wxCoord x
, wxCoord y
)
1128 #ifdef __WXMICROWIN__
1129 if (!GetHDC()) return;
1132 DrawAnyText(text
, x
, y
);
1134 // update the bounding box
1135 CalcBoundingBox(x
, y
);
1138 GetTextExtent(text
, &w
, &h
);
1139 CalcBoundingBox(x
+ w
, y
+ h
);
1142 void wxDC::DrawAnyText(const wxString
& text
, wxCoord x
, wxCoord y
)
1144 #ifdef __WXMICROWIN__
1145 if (!GetHDC()) return;
1148 // prepare for drawing the text
1149 if ( m_textForegroundColour
.Ok() )
1150 SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel());
1152 DWORD old_background
= 0;
1153 if ( m_textBackgroundColour
.Ok() )
1155 old_background
= SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() );
1158 SetBkMode(GetHdc(), m_backgroundMode
== wxTRANSPARENT
? TRANSPARENT
1161 if ( ::TextOut(GetHdc(), XLOG2DEV(x
), YLOG2DEV(y
),
1162 text
.c_str(), text
.length()) == 0 )
1164 wxLogLastError(wxT("TextOut"));
1167 // restore the old parameters (text foreground colour may be left because
1168 // it never is set to anything else, but background should remain
1169 // transparent even if we just drew an opaque string)
1170 if ( m_textBackgroundColour
.Ok() )
1171 (void)SetBkColor(GetHdc(), old_background
);
1173 SetBkMode(GetHdc(), TRANSPARENT
);
1176 void wxDC::DoDrawRotatedText(const wxString
& text
,
1177 wxCoord x
, wxCoord y
,
1180 #ifdef __WXMICROWIN__
1181 if (!GetHDC()) return;
1184 // we test that we have some font because otherwise we should still use the
1185 // "else" part below to avoid that DrawRotatedText(angle = 180) and
1186 // DrawRotatedText(angle = 0) use different fonts (we can't use the default
1187 // font for drawing rotated fonts unfortunately)
1188 if ( (angle
== 0.0) && m_font
.Ok() )
1190 DoDrawText(text
, x
, y
);
1192 #ifndef __WXMICROWIN__
1195 // NB: don't take DEFAULT_GUI_FONT (a.k.a. wxSYS_DEFAULT_GUI_FONT)
1196 // because it's not TrueType and so can't have non zero
1197 // orientation/escapement under Win9x
1198 wxFont font
= m_font
.Ok() ? m_font
: *wxSWISS_FONT
;
1199 HFONT hfont
= (HFONT
)font
.GetResourceHandle();
1201 if ( ::GetObject(hfont
, sizeof(lf
), &lf
) == 0 )
1203 wxLogLastError(wxT("GetObject(hfont)"));
1206 // GDI wants the angle in tenth of degree
1207 long angle10
= (long)(angle
* 10);
1208 lf
.lfEscapement
= angle10
;
1209 lf
. lfOrientation
= angle10
;
1211 hfont
= ::CreateFontIndirect(&lf
);
1214 wxLogLastError(wxT("CreateFont"));
1218 HFONT hfontOld
= (HFONT
)::SelectObject(GetHdc(), hfont
);
1220 DrawAnyText(text
, x
, y
);
1222 (void)::SelectObject(GetHdc(), hfontOld
);
1223 (void)::DeleteObject(hfont
);
1226 // call the bounding box by adding all four vertices of the rectangle
1227 // containing the text to it (simpler and probably not slower than
1228 // determining which of them is really topmost/leftmost/...)
1230 GetTextExtent(text
, &w
, &h
);
1232 double rad
= DegToRad(angle
);
1234 // "upper left" and "upper right"
1235 CalcBoundingBox(x
, y
);
1236 CalcBoundingBox(x
+ wxCoord(w
*cos(rad
)), y
- wxCoord(h
*sin(rad
)));
1238 // "bottom left" and "bottom right"
1239 x
+= (wxCoord
)(h
*sin(rad
));
1240 y
+= (wxCoord
)(h
*cos(rad
));
1241 CalcBoundingBox(x
, y
);
1242 CalcBoundingBox(x
+ wxCoord(h
*sin(rad
)), y
+ wxCoord(h
*cos(rad
)));
1247 // ---------------------------------------------------------------------------
1249 // ---------------------------------------------------------------------------
1253 void wxDC::DoSelectPalette(bool realize
)
1255 #ifdef __WXMICROWIN__
1256 if (!GetHDC()) return;
1259 // Set the old object temporarily, in case the assignment deletes an object
1260 // that's not yet selected out.
1263 ::SelectPalette(GetHdc(), (HPALETTE
) m_oldPalette
, FALSE
);
1267 if ( m_palette
.Ok() )
1269 HPALETTE oldPal
= ::SelectPalette(GetHdc(),
1270 GetHpaletteOf(m_palette
),
1273 m_oldPalette
= (WXHPALETTE
) oldPal
;
1276 ::RealizePalette(GetHdc());
1280 void wxDC::SetPalette(const wxPalette
& palette
)
1284 m_palette
= palette
;
1285 DoSelectPalette(TRUE
);
1289 void wxDC::InitializePalette()
1291 if ( wxDisplayDepth() <= 8 )
1293 // look for any window or parent that has a custom palette. If any has
1294 // one then we need to use it in drawing operations
1295 wxWindow
*win
= m_canvas
->GetAncestorWithCustomPalette();
1297 m_hasCustomPalette
= win
&& win
->HasCustomPalette();
1298 if ( m_hasCustomPalette
)
1300 m_palette
= win
->GetPalette();
1302 // turn on MSW translation for this palette
1308 #endif // wxUSE_PALETTE
1310 void wxDC::SetFont(const wxFont
& the_font
)
1312 #ifdef __WXMICROWIN__
1313 if (!GetHDC()) return;
1316 // Set the old object temporarily, in case the assignment deletes an object
1317 // that's not yet selected out.
1320 ::SelectObject(GetHdc(), (HFONT
) m_oldFont
);
1329 ::SelectObject(GetHdc(), (HFONT
) m_oldFont
);
1333 if (m_font
.Ok() && m_font
.GetResourceHandle())
1335 HFONT f
= (HFONT
) ::SelectObject(GetHdc(), (HFONT
) m_font
.GetResourceHandle());
1336 if (f
== (HFONT
) NULL
)
1338 wxLogDebug(wxT("::SelectObject failed in wxDC::SetFont."));
1341 m_oldFont
= (WXHFONT
) f
;
1345 void wxDC::SetPen(const wxPen
& pen
)
1347 #ifdef __WXMICROWIN__
1348 if (!GetHDC()) return;
1351 // Set the old object temporarily, in case the assignment deletes an object
1352 // that's not yet selected out.
1355 ::SelectObject(GetHdc(), (HPEN
) m_oldPen
);
1364 ::SelectObject(GetHdc(), (HPEN
) m_oldPen
);
1370 if (m_pen
.GetResourceHandle())
1372 HPEN p
= (HPEN
) ::SelectObject(GetHdc(), (HPEN
)m_pen
.GetResourceHandle());
1374 m_oldPen
= (WXHPEN
) p
;
1379 void wxDC::SetBrush(const wxBrush
& brush
)
1381 #ifdef __WXMICROWIN__
1382 if (!GetHDC()) return;
1385 // Set the old object temporarily, in case the assignment deletes an object
1386 // that's not yet selected out.
1389 ::SelectObject(GetHdc(), (HBRUSH
) m_oldBrush
);
1398 ::SelectObject(GetHdc(), (HBRUSH
) m_oldBrush
);
1404 // to make sure the brush is alligned with the logical coordinates
1405 wxBitmap
*stipple
= m_brush
.GetStipple();
1406 if ( stipple
&& stipple
->Ok() )
1409 ::SetBrushOrgEx(GetHdc(),
1410 m_deviceOriginX
% stipple
->GetWidth(),
1411 m_deviceOriginY
% stipple
->GetHeight(),
1412 NULL
); // don't need previous brush origin
1414 ::SetBrushOrg(GetHdc(),
1415 m_deviceOriginX
% stipple
->GetWidth(),
1416 m_deviceOriginY
% stipple
->GetHeight());
1420 if ( m_brush
.GetResourceHandle() )
1423 b
= (HBRUSH
) ::SelectObject(GetHdc(), (HBRUSH
)m_brush
.GetResourceHandle());
1425 m_oldBrush
= (WXHBRUSH
) b
;
1430 void wxDC::SetBackground(const wxBrush
& brush
)
1432 #ifdef __WXMICROWIN__
1433 if (!GetHDC()) return;
1436 m_backgroundBrush
= brush
;
1438 if (!m_backgroundBrush
.Ok())
1443 bool customColours
= TRUE
;
1444 // If we haven't specified wxUSER_COLOURS, don't allow the panel/dialog box to
1445 // change background colours from the control-panel specified colours.
1446 if (m_canvas
->IsKindOf(CLASSINFO(wxWindow
)) && ((m_canvas
->GetWindowStyleFlag() & wxUSER_COLOURS
) != wxUSER_COLOURS
))
1447 customColours
= FALSE
;
1451 if (m_backgroundBrush
.GetStyle()==wxTRANSPARENT
)
1453 m_canvas
->SetTransparent(TRUE
);
1457 // New behaviour, 10/2/99: setting the background brush of a DC
1458 // doesn't affect the window background colour. However,
1459 // I'm leaving in the transparency setting because it's needed by
1460 // various controls (e.g. wxStaticText) to determine whether to draw
1461 // transparently or not. TODO: maybe this should be a new function
1462 // wxWindow::SetTransparency(). Should that apply to the child itself, or the
1464 // m_canvas->SetBackgroundColour(m_backgroundBrush.GetColour());
1465 m_canvas
->SetTransparent(FALSE
);
1469 COLORREF new_color
= m_backgroundBrush
.GetColour().GetPixel();
1471 (void)SetBkColor(GetHdc(), new_color
);
1475 void wxDC::SetBackgroundMode(int mode
)
1477 #ifdef __WXMICROWIN__
1478 if (!GetHDC()) return;
1481 m_backgroundMode
= mode
;
1483 // SetBackgroundColour now only refers to text background
1484 // and m_backgroundMode is used there
1487 void wxDC::SetLogicalFunction(int function
)
1489 #ifdef __WXMICROWIN__
1490 if (!GetHDC()) return;
1493 m_logicalFunction
= function
;
1498 void wxDC::SetRop(WXHDC dc
)
1500 if ( !dc
|| m_logicalFunction
< 0 )
1505 switch (m_logicalFunction
)
1507 case wxCLEAR
: rop
= R2_BLACK
; break;
1508 case wxXOR
: rop
= R2_XORPEN
; break;
1509 case wxINVERT
: rop
= R2_NOT
; break;
1510 case wxOR_REVERSE
: rop
= R2_MERGEPENNOT
; break;
1511 case wxAND_REVERSE
: rop
= R2_MASKPENNOT
; break;
1512 case wxCOPY
: rop
= R2_COPYPEN
; break;
1513 case wxAND
: rop
= R2_MASKPEN
; break;
1514 case wxAND_INVERT
: rop
= R2_MASKNOTPEN
; break;
1515 case wxNO_OP
: rop
= R2_NOP
; break;
1516 case wxNOR
: rop
= R2_NOTMERGEPEN
; break;
1517 case wxEQUIV
: rop
= R2_NOTXORPEN
; break;
1518 case wxSRC_INVERT
: rop
= R2_NOTCOPYPEN
; break;
1519 case wxOR_INVERT
: rop
= R2_MERGENOTPEN
; break;
1520 case wxNAND
: rop
= R2_NOTMASKPEN
; break;
1521 case wxOR
: rop
= R2_MERGEPEN
; break;
1522 case wxSET
: rop
= R2_WHITE
; break;
1525 wxFAIL_MSG( wxT("unsupported logical function") );
1529 SetROP2(GetHdc(), rop
);
1532 bool wxDC::StartDoc(const wxString
& WXUNUSED(message
))
1534 // We might be previewing, so return TRUE to let it continue.
1542 void wxDC::StartPage()
1546 void wxDC::EndPage()
1550 // ---------------------------------------------------------------------------
1552 // ---------------------------------------------------------------------------
1554 wxCoord
wxDC::GetCharHeight() const
1556 #ifdef __WXMICROWIN__
1557 if (!GetHDC()) return 0;
1560 TEXTMETRIC lpTextMetric
;
1562 GetTextMetrics(GetHdc(), &lpTextMetric
);
1564 return lpTextMetric
.tmHeight
;
1567 wxCoord
wxDC::GetCharWidth() const
1569 #ifdef __WXMICROWIN__
1570 if (!GetHDC()) return 0;
1573 TEXTMETRIC lpTextMetric
;
1575 GetTextMetrics(GetHdc(), &lpTextMetric
);
1577 return lpTextMetric
.tmAveCharWidth
;
1580 void wxDC::DoGetTextExtent(const wxString
& string
, wxCoord
*x
, wxCoord
*y
,
1581 wxCoord
*descent
, wxCoord
*externalLeading
,
1584 #ifdef __WXMICROWIN__
1589 if (descent
) *descent
= 0;
1590 if (externalLeading
) *externalLeading
= 0;
1593 #endif // __WXMICROWIN__
1598 wxASSERT_MSG( font
->Ok(), _T("invalid font in wxDC::GetTextExtent") );
1600 hfontOld
= (HFONT
)::SelectObject(GetHdc(), GetHfontOf(*font
));
1602 else // don't change the font
1610 GetTextExtentPoint(GetHdc(), string
, string
.length(), &sizeRect
);
1611 GetTextMetrics(GetHdc(), &tm
);
1618 *descent
= tm
.tmDescent
;
1619 if (externalLeading
)
1620 *externalLeading
= tm
.tmExternalLeading
;
1624 ::SelectObject(GetHdc(), hfontOld
);
1628 void wxDC::SetMapMode(int mode
)
1630 #ifdef __WXMICROWIN__
1631 if (!GetHDC()) return;
1634 m_mappingMode
= mode
;
1636 if ( mode
== wxMM_TEXT
)
1639 m_logicalScaleY
= 1.0;
1641 else // need to do some calculations
1643 int pixel_width
= ::GetDeviceCaps(GetHdc(), HORZRES
),
1644 pixel_height
= ::GetDeviceCaps(GetHdc(), VERTRES
),
1645 mm_width
= ::GetDeviceCaps(GetHdc(), HORZSIZE
),
1646 mm_height
= ::GetDeviceCaps(GetHdc(), VERTSIZE
);
1648 if ( (mm_width
== 0) || (mm_height
== 0) )
1650 // we can't calculate mm2pixels[XY] then!
1654 double mm2pixelsX
= pixel_width
/ mm_width
,
1655 mm2pixelsY
= pixel_height
/ mm_height
;
1660 m_logicalScaleX
= twips2mm
* mm2pixelsX
;
1661 m_logicalScaleY
= twips2mm
* mm2pixelsY
;
1665 m_logicalScaleX
= pt2mm
* mm2pixelsX
;
1666 m_logicalScaleY
= pt2mm
* mm2pixelsY
;
1670 m_logicalScaleX
= mm2pixelsX
;
1671 m_logicalScaleY
= mm2pixelsY
;
1675 m_logicalScaleX
= mm2pixelsX
/ 10.0;
1676 m_logicalScaleY
= mm2pixelsY
/ 10.0;
1680 wxFAIL_MSG( _T("unknown mapping mode in SetMapMode") );
1684 // VZ: it seems very wasteful to always use MM_ANISOTROPIC when in 99% of
1685 // cases we could do with MM_TEXT and in the remaining 0.9% with
1686 // MM_ISOTROPIC (TODO!)
1687 ::SetMapMode(GetHdc(), MM_ANISOTROPIC
);
1689 int width
= DeviceToLogicalXRel(VIEWPORT_EXTENT
)*m_signX
,
1690 height
= DeviceToLogicalYRel(VIEWPORT_EXTENT
)*m_signY
;
1692 ::SetViewportExtEx(GetHdc(), VIEWPORT_EXTENT
, VIEWPORT_EXTENT
, NULL
);
1693 ::SetWindowExtEx(GetHdc(), width
, height
, NULL
);
1695 ::SetViewportOrgEx(GetHdc(), m_deviceOriginX
, m_deviceOriginY
, NULL
);
1696 ::SetWindowOrgEx(GetHdc(), m_logicalOriginX
, m_logicalOriginY
, NULL
);
1699 void wxDC::SetUserScale(double x
, double y
)
1701 #ifdef __WXMICROWIN__
1702 if (!GetHDC()) return;
1705 if ( x
== m_userScaleX
&& y
== m_userScaleY
)
1711 SetMapMode(m_mappingMode
);
1714 void wxDC::SetAxisOrientation(bool xLeftRight
, bool yBottomUp
)
1716 #ifdef __WXMICROWIN__
1717 if (!GetHDC()) return;
1720 int signX
= xLeftRight
? 1 : -1,
1721 signY
= yBottomUp
? -1 : 1;
1723 if ( signX
!= m_signX
|| signY
!= m_signY
)
1728 SetMapMode(m_mappingMode
);
1732 void wxDC::SetSystemScale(double x
, double y
)
1734 #ifdef __WXMICROWIN__
1735 if (!GetHDC()) return;
1738 if ( x
== m_scaleX
&& y
== m_scaleY
)
1744 SetMapMode(m_mappingMode
);
1747 void wxDC::SetLogicalOrigin(wxCoord x
, wxCoord y
)
1749 #ifdef __WXMICROWIN__
1750 if (!GetHDC()) return;
1753 if ( x
== m_logicalOriginX
&& y
== m_logicalOriginY
)
1756 m_logicalOriginX
= x
;
1757 m_logicalOriginY
= y
;
1759 ::SetWindowOrgEx(GetHdc(), (int)m_logicalOriginX
, (int)m_logicalOriginY
, NULL
);
1762 void wxDC::SetDeviceOrigin(wxCoord x
, wxCoord y
)
1764 #ifdef __WXMICROWIN__
1765 if (!GetHDC()) return;
1768 if ( x
== m_deviceOriginX
&& y
== m_deviceOriginY
)
1771 m_deviceOriginX
= x
;
1772 m_deviceOriginY
= y
;
1774 ::SetViewportOrgEx(GetHdc(), (int)m_deviceOriginX
, (int)m_deviceOriginY
, NULL
);
1777 // ---------------------------------------------------------------------------
1778 // coordinates transformations
1779 // ---------------------------------------------------------------------------
1781 wxCoord
wxDCBase::DeviceToLogicalX(wxCoord x
) const
1783 return DeviceToLogicalXRel(x
- m_deviceOriginX
)*m_signX
+ m_logicalOriginX
;
1786 wxCoord
wxDCBase::DeviceToLogicalXRel(wxCoord x
) const
1788 // axis orientation is not taken into account for conversion of a distance
1789 return (wxCoord
)(x
/ (m_logicalScaleX
*m_userScaleX
*m_scaleX
));
1792 wxCoord
wxDCBase::DeviceToLogicalY(wxCoord y
) const
1794 return DeviceToLogicalYRel(y
- m_deviceOriginY
)*m_signY
+ m_logicalOriginY
;
1797 wxCoord
wxDCBase::DeviceToLogicalYRel(wxCoord y
) const
1799 // axis orientation is not taken into account for conversion of a distance
1800 return (wxCoord
)( y
/ (m_logicalScaleY
*m_userScaleY
*m_scaleY
));
1803 wxCoord
wxDCBase::LogicalToDeviceX(wxCoord x
) const
1805 return LogicalToDeviceXRel(x
- m_logicalOriginX
)*m_signX
+ m_deviceOriginX
;
1808 wxCoord
wxDCBase::LogicalToDeviceXRel(wxCoord x
) const
1810 // axis orientation is not taken into account for conversion of a distance
1811 return (wxCoord
) (x
*m_logicalScaleX
*m_userScaleX
*m_scaleX
);
1814 wxCoord
wxDCBase::LogicalToDeviceY(wxCoord y
) const
1816 return LogicalToDeviceYRel(y
- m_logicalOriginY
)*m_signY
+ m_deviceOriginY
;
1819 wxCoord
wxDCBase::LogicalToDeviceYRel(wxCoord y
) const
1821 // axis orientation is not taken into account for conversion of a distance
1822 return (wxCoord
) (y
*m_logicalScaleY
*m_userScaleY
*m_scaleY
);
1825 // ---------------------------------------------------------------------------
1827 // ---------------------------------------------------------------------------
1829 bool wxDC::DoBlit(wxCoord xdest
, wxCoord ydest
,
1830 wxCoord width
, wxCoord height
,
1831 wxDC
*source
, wxCoord xsrc
, wxCoord ysrc
,
1832 int rop
, bool useMask
,
1833 wxCoord xsrcMask
, wxCoord ysrcMask
)
1835 #ifdef __WXMICROWIN__
1836 if (!GetHDC()) return FALSE
;
1839 const wxBitmap
& bmpSrc
= source
->m_selectedBitmap
;
1841 wxMask
*mask
= NULL
;
1844 mask
= bmpSrc
.GetMask();
1846 if ( !(bmpSrc
.Ok() && mask
&& mask
->GetMaskBitmap()) )
1848 // don't give assert here because this would break existing
1849 // programs - just silently ignore useMask parameter
1854 if (xsrcMask
== -1 && ysrcMask
== -1)
1856 xsrcMask
= xsrc
; ysrcMask
= ysrc
;
1859 COLORREF old_textground
= ::GetTextColor(GetHdc());
1860 COLORREF old_background
= ::GetBkColor(GetHdc());
1861 if (m_textForegroundColour
.Ok())
1863 ::SetTextColor(GetHdc(), m_textForegroundColour
.GetPixel() );
1865 if (m_textBackgroundColour
.Ok())
1867 ::SetBkColor(GetHdc(), m_textBackgroundColour
.GetPixel() );
1870 DWORD dwRop
= SRCCOPY
;
1873 case wxXOR
: dwRop
= SRCINVERT
; break;
1874 case wxINVERT
: dwRop
= DSTINVERT
; break;
1875 case wxOR_REVERSE
: dwRop
= 0x00DD0228; break;
1876 case wxAND_REVERSE
: dwRop
= SRCERASE
; break;
1877 case wxCLEAR
: dwRop
= BLACKNESS
; break;
1878 case wxSET
: dwRop
= WHITENESS
; break;
1879 case wxOR_INVERT
: dwRop
= MERGEPAINT
; break;
1880 case wxAND
: dwRop
= SRCAND
; break;
1881 case wxOR
: dwRop
= SRCPAINT
; break;
1882 case wxEQUIV
: dwRop
= 0x00990066; break;
1883 case wxNAND
: dwRop
= 0x007700E6; break;
1884 case wxAND_INVERT
: dwRop
= 0x00220326; break;
1885 case wxCOPY
: dwRop
= SRCCOPY
; break;
1886 case wxNO_OP
: dwRop
= DSTCOPY
; break;
1887 case wxSRC_INVERT
: dwRop
= NOTSRCCOPY
; break;
1888 case wxNOR
: dwRop
= NOTSRCCOPY
; break;
1890 wxFAIL_MSG( wxT("unsupported logical function") );
1894 bool success
= FALSE
;
1899 // we want the part of the image corresponding to the mask to be
1900 // transparent, so use "DSTCOPY" ROP for the mask points (the usual
1901 // meaning of fg and bg is inverted which corresponds to wxWin notion
1902 // of the mask which is also contrary to the Windows one)
1904 // On some systems, MaskBlt succeeds yet is much much slower
1905 // than the wxWindows fall-back implementation. So we need
1906 // to be able to switch this on and off at runtime.
1907 #if wxUSE_SYSTEM_OPTIONS
1908 if (wxSystemOptions::GetOptionInt(wxT("no-maskblt")) == 0)
1914 xdest
, ydest
, width
, height
,
1917 (HBITMAP
)mask
->GetMaskBitmap(),
1919 MAKEROP4(dwRop
, DSTCOPY
)
1926 // Blit bitmap with mask
1929 HBITMAP buffer_bmap
;
1931 #if wxUSE_DC_CACHEING
1932 // create a temp buffer bitmap and DCs to access it and the mask
1933 wxDCCacheEntry
* dcCacheEntry1
= FindDCInCache(NULL
, source
->GetHDC());
1934 dc_mask
= (HDC
) dcCacheEntry1
->m_dc
;
1936 wxDCCacheEntry
* dcCacheEntry2
= FindDCInCache(dcCacheEntry1
, GetHDC());
1937 dc_buffer
= (HDC
) dcCacheEntry2
->m_dc
;
1939 wxDCCacheEntry
* bitmapCacheEntry
= FindBitmapInCache(GetHDC(),
1942 buffer_bmap
= (HBITMAP
) bitmapCacheEntry
->m_bitmap
;
1943 #else // !wxUSE_DC_CACHEING
1944 // create a temp buffer bitmap and DCs to access it and the mask
1945 dc_mask
= ::CreateCompatibleDC(GetHdcOf(*source
));
1946 dc_buffer
= ::CreateCompatibleDC(GetHdc());
1947 buffer_bmap
= ::CreateCompatibleBitmap(GetHdc(), width
, height
);
1948 #endif // wxUSE_DC_CACHEING/!wxUSE_DC_CACHEING
1949 HGDIOBJ hOldMaskBitmap
= ::SelectObject(dc_mask
, (HBITMAP
) mask
->GetMaskBitmap());
1950 HGDIOBJ hOldBufferBitmap
= ::SelectObject(dc_buffer
, buffer_bmap
);
1952 // copy dest to buffer
1953 if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
,
1954 GetHdc(), xdest
, ydest
, SRCCOPY
) )
1956 wxLogLastError(wxT("BitBlt"));
1959 // copy src to buffer using selected raster op
1960 if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
,
1961 GetHdcOf(*source
), xsrc
, ysrc
, dwRop
) )
1963 wxLogLastError(wxT("BitBlt"));
1966 // set masked area in buffer to BLACK (pixel value 0)
1967 COLORREF prevBkCol
= ::SetBkColor(GetHdc(), RGB(255, 255, 255));
1968 COLORREF prevCol
= ::SetTextColor(GetHdc(), RGB(0, 0, 0));
1969 if ( !::BitBlt(dc_buffer
, 0, 0, (int)width
, (int)height
,
1970 dc_mask
, xsrcMask
, ysrcMask
, SRCAND
) )
1972 wxLogLastError(wxT("BitBlt"));
1975 // set unmasked area in dest to BLACK
1976 ::SetBkColor(GetHdc(), RGB(0, 0, 0));
1977 ::SetTextColor(GetHdc(), RGB(255, 255, 255));
1978 if ( !::BitBlt(GetHdc(), xdest
, ydest
, (int)width
, (int)height
,
1979 dc_mask
, xsrcMask
, ysrcMask
, SRCAND
) )
1981 wxLogLastError(wxT("BitBlt"));
1983 ::SetBkColor(GetHdc(), prevBkCol
); // restore colours to original values
1984 ::SetTextColor(GetHdc(), prevCol
);
1986 // OR buffer to dest
1987 success
= ::BitBlt(GetHdc(), xdest
, ydest
,
1988 (int)width
, (int)height
,
1989 dc_buffer
, 0, 0, SRCPAINT
) != 0;
1992 wxLogLastError(wxT("BitBlt"));
1995 // tidy up temporary DCs and bitmap
1996 ::SelectObject(dc_mask
, hOldMaskBitmap
);
1997 ::SelectObject(dc_buffer
, hOldBufferBitmap
);
1999 #if !wxUSE_DC_CACHEING
2001 ::DeleteDC(dc_mask
);
2002 ::DeleteDC(dc_buffer
);
2003 ::DeleteObject(buffer_bmap
);
2008 else // no mask, just BitBlt() it
2010 // if we already have a DIB, draw it using StretchDIBits(), otherwise
2011 // use StretchBlt() if available and finally fall back to BitBlt()
2012 const int caps
= ::GetDeviceCaps(GetHdc(), RASTERCAPS
);
2013 if ( bmpSrc
.Ok() && (caps
& RC_STRETCHDIB
) )
2018 if ( ::GetObject(GetHbitmapOf(bmpSrc
),
2020 &ds
) == sizeof(ds
) )
2022 StretchBltModeChanger
changeMode(GetHdc(), COLORONCOLOR
);
2024 if ( ::StretchDIBits(GetHdc(),
2030 (LPBITMAPINFO
)&ds
.dsBmih
,
2033 ) == (int)GDI_ERROR
)
2035 wxLogLastError(wxT("StretchDIBits"));
2044 if ( !success
&& (caps
& RC_STRETCHBLT
) )
2046 StretchBltModeChanger
changeMode(GetHdc(), COLORONCOLOR
);
2051 xdest
, ydest
, width
, height
,
2053 xsrc
, ysrc
, width
, height
,
2057 wxLogLastError(_T("StretchBlt"));
2071 (int)width
, (int)height
,
2077 wxLogLastError(_T("BitBlt"));
2086 ::SetTextColor(GetHdc(), old_textground
);
2087 ::SetBkColor(GetHdc(), old_background
);
2092 void wxDC::DoGetSize(int *w
, int *h
) const
2094 #ifdef __WXMICROWIN__
2095 if (!GetHDC()) return;
2098 if ( w
) *w
= ::GetDeviceCaps(GetHdc(), HORZRES
);
2099 if ( h
) *h
= ::GetDeviceCaps(GetHdc(), VERTRES
);
2102 void wxDC::DoGetSizeMM(int *w
, int *h
) const
2104 #ifdef __WXMICROWIN__
2105 if (!GetHDC()) return;
2108 // if we implement it in terms of DoGetSize() instead of directly using the
2109 // results returned by GetDeviceCaps(HORZ/VERTSIZE) as was done before, it
2110 // will also work for wxWindowDC and wxClientDC even though their size is
2111 // not the same as the total size of the screen
2112 int wPixels
, hPixels
;
2113 DoGetSize(&wPixels
, &hPixels
);
2117 int wTotal
= ::GetDeviceCaps(GetHdc(), HORZRES
);
2119 wxCHECK_RET( wTotal
, _T("0 width device?") );
2121 *w
= (wPixels
* ::GetDeviceCaps(GetHdc(), HORZSIZE
)) / wTotal
;
2126 int hTotal
= ::GetDeviceCaps(GetHdc(), VERTRES
);
2128 wxCHECK_RET( hTotal
, _T("0 height device?") );
2130 *h
= (hPixels
* ::GetDeviceCaps(GetHdc(), VERTSIZE
)) / hTotal
;
2134 wxSize
wxDC::GetPPI() const
2136 #ifdef __WXMICROWIN__
2137 if (!GetHDC()) return wxSize();
2140 int x
= ::GetDeviceCaps(GetHdc(), LOGPIXELSX
);
2141 int y
= ::GetDeviceCaps(GetHdc(), LOGPIXELSY
);
2143 return wxSize(x
, y
);
2146 // For use by wxWindows only, unless custom units are required.
2147 void wxDC::SetLogicalScale(double x
, double y
)
2149 #ifdef __WXMICROWIN__
2150 if (!GetHDC()) return;
2153 m_logicalScaleX
= x
;
2154 m_logicalScaleY
= y
;
2157 // ----------------------------------------------------------------------------
2159 // ----------------------------------------------------------------------------
2161 #if wxUSE_DC_CACHEING
2164 * This implementation is a bit ugly and uses the old-fashioned wxList class, so I will
2165 * improve it in due course, either using arrays, or simply storing pointers to one
2166 * entry for the bitmap, and two for the DCs. -- JACS
2169 wxList
wxDC::sm_bitmapCache
;
2170 wxList
wxDC::sm_dcCache
;
2172 wxDCCacheEntry::wxDCCacheEntry(WXHBITMAP hBitmap
, int w
, int h
, int depth
)
2181 wxDCCacheEntry::wxDCCacheEntry(WXHDC hDC
, int depth
)
2190 wxDCCacheEntry::~wxDCCacheEntry()
2193 ::DeleteObject((HBITMAP
) m_bitmap
);
2195 ::DeleteDC((HDC
) m_dc
);
2198 wxDCCacheEntry
* wxDC::FindBitmapInCache(WXHDC dc
, int w
, int h
)
2200 int depth
= ::GetDeviceCaps((HDC
) dc
, PLANES
) * ::GetDeviceCaps((HDC
) dc
, BITSPIXEL
);
2201 wxNode
* node
= sm_bitmapCache
.GetFirst();
2204 wxDCCacheEntry
* entry
= (wxDCCacheEntry
*) node
->GetData();
2206 if (entry
->m_depth
== depth
)
2208 if (entry
->m_width
< w
|| entry
->m_height
< h
)
2210 ::DeleteObject((HBITMAP
) entry
->m_bitmap
);
2211 entry
->m_bitmap
= (WXHBITMAP
) ::CreateCompatibleBitmap((HDC
) dc
, w
, h
);
2212 if ( !entry
->m_bitmap
)
2214 wxLogLastError(wxT("CreateCompatibleBitmap"));
2216 entry
->m_width
= w
; entry
->m_height
= h
;
2222 node
= node
->GetNext();
2224 WXHBITMAP hBitmap
= (WXHBITMAP
) ::CreateCompatibleBitmap((HDC
) dc
, w
, h
);
2227 wxLogLastError(wxT("CreateCompatibleBitmap"));
2229 wxDCCacheEntry
* entry
= new wxDCCacheEntry(hBitmap
, w
, h
, depth
);
2230 AddToBitmapCache(entry
);
2234 wxDCCacheEntry
* wxDC::FindDCInCache(wxDCCacheEntry
* notThis
, WXHDC dc
)
2236 int depth
= ::GetDeviceCaps((HDC
) dc
, PLANES
) * ::GetDeviceCaps((HDC
) dc
, BITSPIXEL
);
2237 wxNode
* node
= sm_dcCache
.GetFirst();
2240 wxDCCacheEntry
* entry
= (wxDCCacheEntry
*) node
->GetData();
2242 // Don't return the same one as we already have
2243 if (!notThis
|| (notThis
!= entry
))
2245 if (entry
->m_depth
== depth
)
2251 node
= node
->GetNext();
2253 WXHDC hDC
= (WXHDC
) ::CreateCompatibleDC((HDC
) dc
);
2256 wxLogLastError(wxT("CreateCompatibleDC"));
2258 wxDCCacheEntry
* entry
= new wxDCCacheEntry(hDC
, depth
);
2259 AddToDCCache(entry
);
2263 void wxDC::AddToBitmapCache(wxDCCacheEntry
* entry
)
2265 sm_bitmapCache
.Append(entry
);
2268 void wxDC::AddToDCCache(wxDCCacheEntry
* entry
)
2270 sm_dcCache
.Append(entry
);
2273 void wxDC::ClearCache()
2275 sm_dcCache
.DeleteContents(TRUE
);
2277 sm_dcCache
.DeleteContents(FALSE
);
2278 sm_bitmapCache
.DeleteContents(TRUE
);
2279 sm_bitmapCache
.Clear();
2280 sm_bitmapCache
.DeleteContents(FALSE
);
2283 // Clean up cache at app exit
2284 class wxDCModule
: public wxModule
2287 virtual bool OnInit() { return TRUE
; }
2288 virtual void OnExit() { wxDC::ClearCache(); }
2291 DECLARE_DYNAMIC_CLASS(wxDCModule
)
2294 IMPLEMENT_DYNAMIC_CLASS(wxDCModule
, wxModule
)
2296 #endif // wxUSE_DC_CACHEING
2298 // ----------------------------------------------------------------------------
2299 // wxAlphaBlend: our fallback if ::AlphaBlend() is unavailable
2300 // ----------------------------------------------------------------------------
2303 wxAlphaBlend(wxDC
& dc
, int xDst
, int yDst
, int w
, int h
, const wxBitmap
& bmpSrc
)
2305 // get the destination DC pixels
2306 wxBitmap
bmpDst(w
, h
, 32);
2308 SelectInHDC
select(hdcMem
, GetHbitmapOf(bmpDst
));
2310 if ( !::BitBlt(hdcMem
, 0, 0, w
, h
, GetHdcOf(dc
), 0, 0, SRCCOPY
) )
2312 wxLogLastError(_T("BitBlt"));
2315 // combine them with the source bitmap using alpha
2316 wxRawBitmapData
dataDst(bmpDst
),
2319 wxRawBitmapIterator
pDst(dataDst
),
2322 for ( int y
= 0; y
< h
; y
++ )
2324 wxRawBitmapIterator pDstRowStart
= pDst
,
2325 pSrcRowStart
= pSrc
;
2327 for ( int x
= 0; x
< w
; x
++ )
2329 // note that source bitmap uses premultiplied alpha (as required by
2330 // the real AlphaBlend)
2331 const unsigned beta
= 255 - pSrc
.Alpha();
2333 pDst
.Red() = pSrc
.Red() + (beta
* pDst
.Red() + 127) / 255;
2334 pDst
.Blue() = pSrc
.Blue() + (beta
* pDst
.Blue() + 127) / 255;
2335 pDst
.Green() = pSrc
.Green() + (beta
* pDst
.Green() + 127) / 255;
2341 pDst
= pDstRowStart
;
2342 pSrc
= pSrcRowStart
;
2347 // and finally blit them back to the destination DC
2348 if ( !::BitBlt(GetHdcOf(dc
), xDst
, yDst
, w
, h
, hdcMem
, 0, 0, SRCCOPY
) )
2350 wxLogLastError(_T("BitBlt"));