]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/dcprint.cpp
8105eec4413e8a4ed55a9ed49d04b0443554db28
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/dcprint.cpp
3 // Purpose: wxPrinterDC class
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "dcprint.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
32 #include "wx/string.h"
34 #include "wx/window.h"
37 #include "wx/msw/private.h"
38 #include "wx/dcprint.h"
41 #if wxUSE_COMMON_DIALOGS || defined(__WXWINE__)
49 // ----------------------------------------------------------------------------
51 // ----------------------------------------------------------------------------
52 IMPLEMENT_CLASS(wxPrinterDC
, wxDC
)
54 // ============================================================================
56 // ============================================================================
58 // ----------------------------------------------------------------------------
59 // wxPrinterDC construction
60 // ----------------------------------------------------------------------------
62 // This form is deprecated
63 wxPrinterDC::wxPrinterDC(const wxString
& driver_name
, const wxString
& device_name
, const wxString
& file
, bool interactive
, int orientation
)
65 m_isInteractive
= interactive
;
68 m_printData
.SetFilename(file
);
70 #if wxUSE_COMMON_DIALOGS
75 pd
.lStructSize
= sizeof( PRINTDLG
);
76 pd
.hwndOwner
=(HWND
) NULL
;
77 pd
.hDevMode
=(HANDLE
)NULL
;
78 pd
.hDevNames
=(HANDLE
)NULL
;
79 pd
.Flags
=PD_RETURNDC
| PD_NOSELECTION
| PD_NOPAGENUMS
;
85 pd
.hInstance
=(HINSTANCE
)NULL
;
87 if ( PrintDlg( &pd
) != 0 )
89 m_hDC
= (WXHDC
) pd
.hDC
;
98 // m_dontDelete = TRUE;
101 #endif // wxUSE_COMMON_DIALOGS
102 if ((!driver_name
.IsNull() && driver_name
!= wxT("")) &&
103 (!device_name
.IsNull() && device_name
!= wxT("")) &&
104 (!file
.IsNull() && file
!= wxT("")))
106 m_hDC
= (WXHDC
) CreateDC(WXSTRINGCAST driver_name
, WXSTRINGCAST device_name
, WXSTRINGCAST file
, NULL
);
107 m_ok
= m_hDC
? TRUE
: FALSE
;
111 wxPrintData printData
;
112 printData
.SetOrientation(orientation
);
113 m_hDC
= wxGetPrinterDC(printData
);
114 m_ok
= m_hDC
? TRUE
: FALSE
;
119 // int width = GetDeviceCaps(m_hDC, VERTRES);
120 // int height = GetDeviceCaps(m_hDC, HORZRES);
121 SetMapMode(wxMM_TEXT
);
123 SetBrush(*wxBLACK_BRUSH
);
124 SetPen(*wxBLACK_PEN
);
127 wxPrinterDC::wxPrinterDC(const wxPrintData
& printData
)
129 m_printData
= printData
;
131 m_isInteractive
= FALSE
;
133 m_hDC
= wxGetPrinterDC(printData
);
137 SetMapMode(wxMM_TEXT
);
139 SetBrush(*wxBLACK_BRUSH
);
140 SetPen(*wxBLACK_PEN
);
144 wxPrinterDC::wxPrinterDC(WXHDC theDC
)
146 m_isInteractive
= FALSE
;
152 // int width = GetDeviceCaps(m_hDC, VERTRES);
153 // int height = GetDeviceCaps(m_hDC, HORZRES);
154 SetMapMode(wxMM_TEXT
);
156 SetBrush(*wxBLACK_BRUSH
);
157 SetPen(*wxBLACK_PEN
);
160 wxPrinterDC::~wxPrinterDC()
164 // ----------------------------------------------------------------------------
165 // wxPrinterDC {Start/End}{Page/Doc} methods
166 // ----------------------------------------------------------------------------
168 bool wxPrinterDC::StartDoc(const wxString
& message
)
171 docinfo
.cbSize
= sizeof(DOCINFO
);
172 docinfo
.lpszDocName
= (const wxChar
*)message
;
174 wxString
filename(m_printData
.GetFilename());
176 if (filename
.IsEmpty())
177 docinfo
.lpszOutput
= NULL
;
179 docinfo
.lpszOutput
= (const wxChar
*) filename
;
181 #if defined(__WIN95__)
182 docinfo
.lpszDatatype
= NULL
;
189 int ret
= ::StartDoc(GetHdc(), &docinfo
);
194 DWORD lastError
= GetLastError();
195 wxLogDebug(wxT("wxDC::StartDoc failed with error: %d\n"), lastError
);
202 void wxPrinterDC::EndDoc()
204 if (m_hDC
) ::EndDoc((HDC
) m_hDC
);
207 void wxPrinterDC::StartPage()
210 ::StartPage((HDC
) m_hDC
);
213 void wxPrinterDC::EndPage()
216 ::EndPage((HDC
) m_hDC
);
219 // Returns default device and port names
220 static bool wxGetDefaultDeviceName(wxString
& deviceName
, wxString
& portName
)
224 LPDEVNAMES lpDevNames
;
225 LPSTR lpszDriverName
;
226 LPSTR lpszDeviceName
;
231 // Cygwin has trouble believing PRINTDLG is 66 bytes - thinks it is 68
233 pd
.lStructSize
= 66; // sizeof(PRINTDLG);
235 pd
.lStructSize
= sizeof(PRINTDLG
);
238 pd
.hwndOwner
= (HWND
)NULL
;
239 pd
.hDevMode
= NULL
; // Will be created by PrintDlg
240 pd
.hDevNames
= NULL
; // Ditto
241 pd
.Flags
= PD_RETURNDEFAULT
;
244 if (!PrintDlg((LPPRINTDLG
)&pd
))
247 GlobalFree(pd
.hDevMode
);
249 GlobalFree(pd
.hDevNames
);
256 lpDevNames
= (LPDEVNAMES
)GlobalLock(pd
.hDevNames
);
257 lpszDriverName
= (LPSTR
)lpDevNames
+ lpDevNames
->wDriverOffset
;
258 lpszDeviceName
= (LPSTR
)lpDevNames
+ lpDevNames
->wDeviceOffset
;
259 lpszPortName
= (LPSTR
)lpDevNames
+ lpDevNames
->wOutputOffset
;
261 deviceName
= lpszDeviceName
;
262 portName
= lpszPortName
;
264 GlobalUnlock(pd
.hDevNames
);
265 GlobalFree(pd
.hDevNames
);
271 GlobalFree(pd
.hDevMode
);
274 return ( deviceName
!= wxT("") );
278 // This uses defaults, except for orientation, so we should eliminate this function
279 // and use the 2nd form (passing wxPrintData) instead.
280 WXHDC
wxGetPrinterDC(int orientation
)
283 LPDEVMODE lpDevMode
= NULL
;
284 LPDEVNAMES lpDevNames
;
285 LPSTR lpszDriverName
;
286 LPSTR lpszDeviceName
;
290 // __GNUWIN32__ has trouble believing PRINTDLG is 66 bytes - thinks it is 68
292 pd
.lStructSize
= 66; // sizeof(PRINTDLG);
294 pd
.lStructSize
= sizeof(PRINTDLG
);
296 pd
.hwndOwner
= (HWND
)NULL
;
297 pd
.hDevMode
= NULL
; // Will be created by PrintDlg
298 pd
.hDevNames
= NULL
; // Ditto
299 pd
.Flags
= PD_RETURNDEFAULT
;
302 if (!PrintDlg((LPPRINTDLG
)&pd
))
305 GlobalFree(pd
.hDevMode
);
307 GlobalFree(pd
.hDevNames
);
315 GlobalFree(pd
.hDevMode
);
318 lpDevNames
= (LPDEVNAMES
)GlobalLock(pd
.hDevNames
);
319 lpszDriverName
= (LPSTR
)lpDevNames
+ lpDevNames
->wDriverOffset
;
320 lpszDeviceName
= (LPSTR
)lpDevNames
+ lpDevNames
->wDeviceOffset
;
321 lpszPortName
= (LPSTR
)lpDevNames
+ lpDevNames
->wOutputOffset
;
322 GlobalUnlock(pd
.hDevNames
);
326 lpDevMode
= (DEVMODE
*) GlobalLock(pd
.hDevMode
);
327 lpDevMode
->dmOrientation
= orientation
;
328 lpDevMode
->dmFields
|= DM_ORIENTATION
;
332 hDC
= CreateDC(lpszDriverName
, lpszDeviceName
, lpszPortName
, (DEVMODE
*)lpDevMode
);
334 hDC
= CreateDC(lpszDriverName
, lpszDeviceName
, lpszPortName
, (LPSTR
)lpDevMode
);
337 if (pd
.hDevMode
&& lpDevMode
)
338 GlobalUnlock(pd
.hDevMode
);
342 GlobalFree(pd
.hDevNames
);
347 GlobalFree(pd
.hDevMode
);
354 // Gets an HDC for the specified printer configuration
355 WXHDC WXDLLEXPORT
wxGetPrinterDC(const wxPrintData
& printDataConst
)
357 wxPrintData printData
= printDataConst
;
358 printData
.ConvertToNative();
360 wxChar
* driverName
= (wxChar
*) NULL
;
362 wxString devNameStr
= printData
.GetPrinterName();
363 wxChar
* portName
= (wxChar
*) NULL
; // Obsolete in WIN32
365 const wxChar
* deviceName
;
367 deviceName
= (wxChar
*) NULL
;
369 deviceName
= devNameStr
.c_str();
371 LPDEVMODE lpDevMode
= (LPDEVMODE
) NULL
;
373 HGLOBAL hDevMode
= (HGLOBAL
)(DWORD
) printData
.GetNativeData();
376 lpDevMode
= (DEVMODE
*) GlobalLock(hDevMode
);
380 // Retrieve the default device name
386 #endif // Debug/Release
387 wxGetDefaultDeviceName(devNameStr
, portName
);
389 wxASSERT_MSG( ret
, wxT("Could not get default device name.") );
391 deviceName
= devNameStr
.c_str();
395 HDC hDC
= CreateDC(driverName
, deviceName
, portName
, (DEVMODE
*) lpDevMode
);
397 HDC hDC
= CreateDC(driverName
, deviceName
, portName
, (LPSTR
) lpDevMode
);
400 if (hDevMode
&& lpDevMode
)
401 GlobalUnlock(hDevMode
);
406 // ----------------------------------------------------------------------------
407 // wxPrinterDC bit blitting/bitmap drawing
408 // ----------------------------------------------------------------------------
410 void wxPrinterDC::DoDrawBitmap(const wxBitmap
&bmp
,
411 wxCoord x
, wxCoord y
,
414 wxCHECK_RET( bmp
.Ok(), _T("invalid bitmap in wxPrinterDC::DrawBitmap") );
416 int width
= bmp
.GetWidth(),
417 height
= bmp
.GetHeight();
419 if ( ::GetDeviceCaps(GetHdc(), RASTERCAPS
) & RC_STRETCHDIB
)
421 BITMAPINFO
*info
= (BITMAPINFO
*) malloc( sizeof( BITMAPINFOHEADER
) + 256 * sizeof(RGBQUAD
) );
422 memset( info
, 0, sizeof( BITMAPINFOHEADER
) );
424 int iBitsSize
= ((width
+ 3 ) & ~3 ) * height
;
426 void* bits
= malloc( iBitsSize
);
428 info
->bmiHeader
.biSize
= sizeof( BITMAPINFOHEADER
);
429 info
->bmiHeader
.biWidth
= width
;
430 info
->bmiHeader
.biHeight
= height
;
431 info
->bmiHeader
.biPlanes
= 1;
432 info
->bmiHeader
.biBitCount
= 8;
433 info
->bmiHeader
.biCompression
= BI_RGB
;
436 if ( GetDIBits(display
, GetHbitmapOf(bmp
), 0,
437 bmp
.GetHeight(), bits
, info
,
440 if ( ::StretchDIBits(GetHdc(), x
, y
,
442 0 , 0, width
, height
,
444 DIB_RGB_COLORS
, SRCCOPY
) == GDI_ERROR
)
446 wxLogLastError("StretchDIBits");
453 else // no support for StretchDIBits()
456 memDC
.SelectObject(bmp
);
458 Blit(x
, y
, width
, height
, &memDC
, 0, 0, wxCOPY
, useMask
);
460 memDC
.SelectObject(wxNullBitmap
);
464 bool wxPrinterDC::DoBlit(wxCoord xdest
, wxCoord ydest
,
465 wxCoord width
, wxCoord height
,
467 wxCoord xsrc
, wxCoord ysrc
,
468 int rop
, bool useMask
)
474 // If we are printing source colours are screen colours
475 // not printer colours and so we need copy the bitmap
478 HDC dc_src
= GetHdcOf(*source
);
479 HDC dc_mask
= ::CreateCompatibleDC(dc_src
);
481 ::SelectObject(dc_mask
, (HBITMAP
) source
->GetSelectedBitmap().GetMask()->GetMaskBitmap());
482 for (int x
= 0; x
< width
; x
++)
484 for (int y
= 0; y
< height
; y
++)
486 COLORREF cref
= ::GetPixel(dc_mask
, x
, y
);
489 HBRUSH brush
= ::CreateSolidBrush(::GetPixel(dc_src
, x
, y
));
490 rect
.left
= xdest
+ x
;
491 rect
.right
= rect
.left
+ 1;
492 rect
.top
= ydest
+ y
;
493 rect
.bottom
= rect
.top
+ 1;
494 ::FillRect(GetHdc(), &rect
, brush
);
495 ::DeleteObject(brush
);
499 ::SelectObject(dc_mask
, 0);
504 if ( ::GetDeviceCaps(GetHdc(), RASTERCAPS
) & RC_STRETCHDIB
)
506 wxBitmap
& bmp
= source
->GetSelectedBitmap();
507 int width
= bmp
.GetWidth(),
508 height
= bmp
.GetHeight();
510 BITMAPINFO
*info
= (BITMAPINFO
*) malloc( sizeof( BITMAPINFOHEADER
) + 256 * sizeof(RGBQUAD
) );
511 int iBitsSize
= ((width
+ 3 ) & ~3 ) * height
;
513 void* bits
= malloc( iBitsSize
);
515 memset( info
, 0 , sizeof( BITMAPINFOHEADER
) );
517 info
->bmiHeader
.biSize
= sizeof( BITMAPINFOHEADER
);
518 info
->bmiHeader
.biWidth
= width
;
519 info
->bmiHeader
.biHeight
= height
;
520 info
->bmiHeader
.biPlanes
= 1;
521 info
->bmiHeader
.biBitCount
= 8;
522 info
->bmiHeader
.biCompression
= BI_RGB
;
525 if ( !::GetDIBits(display
, GetHbitmapOf(bmp
), 0,
526 height
, bits
, info
, DIB_RGB_COLORS
) )
528 wxLogLastError("GetDIBits");
535 success
= ::StretchDIBits(GetHdc(), xdest
, ydest
,
541 SRCCOPY
) != GDI_ERROR
;
544 wxLogLastError("StretchDIBits");
551 else // no support for StretchDIBits
553 // as we are printing, source colours are screen colours not printer
554 // colours and so we need copy the bitmap pixel by pixel.
555 HDC dc_src
= GetHdcOf(*source
);
557 for (int y
= 0; y
< height
; y
++)
559 // This is Stefan Csomor's optimisation, where identical adjacent
560 // pixels are drawn together.
561 for (int x
= 0; x
< width
; x
++)
563 COLORREF col
= ::GetPixel(dc_src
, x
, y
);
564 HBRUSH brush
= ::CreateSolidBrush( col
);
566 rect
.left
= xdest
+ x
;
567 rect
.top
= ydest
+ y
;
568 while( (x
+ 1 < width
) && (::GetPixel(dc_src
, x
+ 1, y
) == col
) )
572 rect
.right
= xdest
+ x
+ 1;
573 rect
.bottom
= rect
.top
+ 1;
574 ::FillRect((HDC
) m_hDC
, &rect
, brush
);
575 ::DeleteObject(brush
);