]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/dcprint.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/dcprint.cpp
3 // Purpose: wxPrinterDC class
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
27 #if wxUSE_PRINTING_ARCHITECTURE
29 #include "wx/dcprint.h"
32 #include "wx/msw/wrapcdlg.h"
33 #include "wx/string.h"
35 #include "wx/window.h"
36 #include "wx/dcmemory.h"
40 #include "wx/msw/private.h"
43 #include "wx/msw/dib.h"
46 #include "wx/printdlg.h"
47 #include "wx/msw/printdlg.h"
53 // mingw32 defines GDI_ERROR incorrectly
54 #if defined(__GNUWIN32__) || !defined(GDI_ERROR)
56 #define GDI_ERROR ((int)-1)
59 #if defined(__WXUNIVERSAL__) && wxUSE_POSTSCRIPT_ARCHITECTURE_IN_MSW
60 #define wxUSE_PS_PRINTING 1
62 #define wxUSE_PS_PRINTING 0
65 // ----------------------------------------------------------------------------
67 // ----------------------------------------------------------------------------
69 IMPLEMENT_CLASS(wxPrinterDC
, wxDC
)
71 // ============================================================================
73 // ============================================================================
75 // ----------------------------------------------------------------------------
76 // wxPrinterDC construction
77 // ----------------------------------------------------------------------------
79 // This form is deprecated
80 wxPrinterDC::wxPrinterDC(const wxString
& driver_name
,
81 const wxString
& device_name
,
86 m_isInteractive
= interactive
;
89 m_printData
.SetFilename(file
);
91 #if wxUSE_COMMON_DIALOGS
96 pd
.lStructSize
= sizeof( PRINTDLG
);
97 pd
.hwndOwner
= (HWND
) NULL
;
98 pd
.hDevMode
= (HANDLE
)NULL
;
99 pd
.hDevNames
= (HANDLE
)NULL
;
100 pd
.Flags
= PD_RETURNDC
| PD_NOSELECTION
| PD_NOPAGENUMS
;
106 pd
.hInstance
= (HINSTANCE
)NULL
;
108 m_ok
= PrintDlg( &pd
) != 0;
111 m_hDC
= (WXHDC
) pd
.hDC
;
115 #endif // wxUSE_COMMON_DIALOGS
117 if ( !driver_name
.empty() && !device_name
.empty() && !file
.empty() )
119 m_hDC
= (WXHDC
) CreateDC(driver_name
, device_name
, file
, NULL
);
121 else // we don't have all parameters, ask the user
123 wxPrintData printData
;
124 printData
.SetOrientation(orientation
);
125 m_hDC
= wxGetPrinterDC(printData
);
128 m_ok
= m_hDC
? true: false;
130 // as we created it, we must delete it as well
137 wxPrinterDC::wxPrinterDC(const wxPrintData
& printData
)
139 m_printData
= printData
;
141 m_isInteractive
= false;
143 m_hDC
= wxGetPrinterDC(printData
);
151 wxPrinterDC::wxPrinterDC(WXHDC dc
)
153 m_isInteractive
= false;
160 void wxPrinterDC::Init()
164 // int width = GetDeviceCaps(m_hDC, VERTRES);
165 // int height = GetDeviceCaps(m_hDC, HORZRES);
166 SetMapMode(wxMM_TEXT
);
168 SetBrush(*wxBLACK_BRUSH
);
169 SetPen(*wxBLACK_PEN
);
173 // ----------------------------------------------------------------------------
174 // wxPrinterDC {Start/End}{Page/Doc} methods
175 // ----------------------------------------------------------------------------
177 bool wxPrinterDC::StartDoc(const wxString
& message
)
180 docinfo
.cbSize
= sizeof(DOCINFO
);
181 docinfo
.lpszDocName
= (const wxChar
*)message
;
183 wxString
filename(m_printData
.GetFilename());
185 if (filename
.empty())
186 docinfo
.lpszOutput
= NULL
;
188 docinfo
.lpszOutput
= (const wxChar
*) filename
;
190 docinfo
.lpszDatatype
= NULL
;
196 int ret
= ::StartDoc(GetHdc(), &docinfo
);
200 DWORD lastError
= GetLastError();
201 wxLogDebug(wxT("wxDC::StartDoc failed with error: %ld\n"), lastError
);
207 void wxPrinterDC::EndDoc()
209 if (m_hDC
) ::EndDoc((HDC
) m_hDC
);
212 void wxPrinterDC::StartPage()
215 ::StartPage((HDC
) m_hDC
);
218 void wxPrinterDC::EndPage()
221 ::EndPage((HDC
) m_hDC
);
225 wxRect
wxPrinterDC::GetPaperRect()
228 if (!Ok()) return wxRect(0, 0, 0, 0);
229 int w
= ::GetDeviceCaps((HDC
) m_hDC
, PHYSICALWIDTH
);
230 int h
= ::GetDeviceCaps((HDC
) m_hDC
, PHYSICALHEIGHT
);
231 int x
= -::GetDeviceCaps((HDC
) m_hDC
, PHYSICALOFFSETX
);
232 int y
= -::GetDeviceCaps((HDC
) m_hDC
, PHYSICALOFFSETY
);
233 return wxRect(x
, y
, w
, h
);
237 #if !wxUSE_PS_PRINTING
239 // Returns default device and port names
240 static bool wxGetDefaultDeviceName(wxString
& deviceName
, wxString
& portName
)
244 LPDEVNAMES lpDevNames
;
245 LPTSTR lpszDeviceName
;
250 // Cygwin has trouble believing PRINTDLG is 66 bytes - thinks it is 68
253 pd
.lStructSize
= 66; // sizeof(PRINTDLG);
255 memset(&pd
, 0, sizeof(PRINTDLG
));
256 pd
.lStructSize
= sizeof(PRINTDLG
);
259 pd
.hwndOwner
= (HWND
)NULL
;
260 pd
.hDevMode
= NULL
; // Will be created by PrintDlg
261 pd
.hDevNames
= NULL
; // Ditto
262 pd
.Flags
= PD_RETURNDEFAULT
;
265 if (!PrintDlg((LPPRINTDLG
)&pd
))
268 GlobalFree(pd
.hDevMode
);
270 GlobalFree(pd
.hDevNames
);
277 lpDevNames
= (LPDEVNAMES
)GlobalLock(pd
.hDevNames
);
278 lpszDeviceName
= (LPTSTR
)lpDevNames
+ lpDevNames
->wDeviceOffset
;
279 lpszPortName
= (LPTSTR
)lpDevNames
+ lpDevNames
->wOutputOffset
;
281 deviceName
= lpszDeviceName
;
282 portName
= lpszPortName
;
284 GlobalUnlock(pd
.hDevNames
);
285 GlobalFree(pd
.hDevNames
);
291 GlobalFree(pd
.hDevMode
);
294 return ( !deviceName
.empty() );
297 #endif // !wxUSE_PS_PRINTING
299 // Gets an HDC for the specified printer configuration
300 WXHDC WXDLLEXPORT
wxGetPrinterDC(const wxPrintData
& printDataConst
)
302 #if wxUSE_PS_PRINTING
304 wxUnusedVar(printDataConst
);
306 #else // native Windows printing
307 wxWindowsPrintNativeData
*data
=
308 (wxWindowsPrintNativeData
*) printDataConst
.GetNativeData();
310 data
->TransferFrom( printDataConst
);
312 wxString deviceName
= printDataConst
.GetPrinterName();
313 if ( deviceName
.empty() )
315 // Retrieve the default device name
317 if ( !wxGetDefaultDeviceName(deviceName
, portName
) )
319 return 0; // Could not get default device name
324 HGLOBAL hDevMode
= (HGLOBAL
)(DWORD
) data
->GetDevMode();
326 DEVMODE
*lpDevMode
= hDevMode
? (DEVMODE
*)::GlobalLock(hDevMode
) : NULL
;
328 HDC hDC
= ::CreateDC(NULL
, deviceName
, NULL
, lpDevMode
);
330 wxLogLastError(_T("CreateDC(printer)"));
333 ::GlobalUnlock(hDevMode
);
336 #endif // PostScript/Windows printing
339 // ----------------------------------------------------------------------------
340 // wxPrinterDC bit blitting/bitmap drawing
341 // ----------------------------------------------------------------------------
343 // helper of DoDrawBitmap() and DoBlit()
345 bool DrawBitmapUsingStretchDIBits(HDC hdc
,
347 wxCoord x
, wxCoord y
)
351 bool ok
= dib
.IsOk();
356 if ( !::GetObject(dib
.GetHandle(), sizeof(ds
), &ds
) )
358 wxLogLastError(_T("GetObject(DIBSECTION)"));
363 // ok, we've got all data we need, do blit it
368 ds
.dsBmih
.biWidth
, ds
.dsBmih
.biHeight
,
370 ds
.dsBmih
.biWidth
, ds
.dsBmih
.biHeight
,
372 (LPBITMAPINFO
)&ds
.dsBmih
,
377 wxLogLastError(wxT("StretchDIBits"));
388 void wxPrinterDC::DoDrawBitmap(const wxBitmap
& bmp
,
389 wxCoord x
, wxCoord y
,
392 wxCHECK_RET( bmp
.Ok(), _T("invalid bitmap in wxPrinterDC::DrawBitmap") );
394 int width
= bmp
.GetWidth(),
395 height
= bmp
.GetHeight();
397 if ( !(::GetDeviceCaps(GetHdc(), RASTERCAPS
) & RC_STRETCHDIB
) ||
398 !DrawBitmapUsingStretchDIBits(GetHdc(), bmp
, x
, y
) )
400 // no support for StretchDIBits() or an error occurred if we got here
402 memDC
.SelectObject(bmp
);
404 Blit(x
, y
, width
, height
, &memDC
, 0, 0, wxCOPY
, useMask
);
406 memDC
.SelectObject(wxNullBitmap
);
410 bool wxPrinterDC::DoBlit(wxCoord xdest
, wxCoord ydest
,
411 wxCoord width
, wxCoord height
,
413 wxCoord
WXUNUSED(xsrc
), wxCoord
WXUNUSED(ysrc
),
414 int WXUNUSED(rop
), bool useMask
,
415 wxCoord
WXUNUSED(xsrcMask
), wxCoord
WXUNUSED(ysrcMask
))
417 wxBitmap
& bmp
= source
->GetSelectedBitmap();
418 wxMask
*mask
= useMask
? bmp
.GetMask() : NULL
;
421 // If we are printing source colours are screen colours not printer
422 // colours and so we need copy the bitmap pixel by pixel.
424 HDC dcSrc
= GetHdcOf(*source
);
425 MemoryHDC
dcMask(dcSrc
);
426 SelectInHDC
selectMask(dcMask
, (HBITMAP
)mask
->GetMaskBitmap());
428 for (int x
= 0; x
< width
; x
++)
430 for (int y
= 0; y
< height
; y
++)
432 COLORREF cref
= ::GetPixel(dcMask
, x
, y
);
435 HBRUSH brush
= ::CreateSolidBrush(::GetPixel(dcSrc
, x
, y
));
436 rect
.left
= xdest
+ x
;
437 rect
.right
= rect
.left
+ 1;
438 rect
.top
= ydest
+ y
;
439 rect
.bottom
= rect
.top
+ 1;
440 ::FillRect(GetHdc(), &rect
, brush
);
441 ::DeleteObject(brush
);
448 if ( !(::GetDeviceCaps(GetHdc(), RASTERCAPS
) & RC_STRETCHDIB
) ||
449 !DrawBitmapUsingStretchDIBits(GetHdc(), bmp
, xdest
, ydest
) )
451 // no support for StretchDIBits
453 // as we are printing, source colours are screen colours not
454 // printer colours and so we need copy the bitmap pixel by pixel.
455 HDC dcSrc
= GetHdcOf(*source
);
457 for (int y
= 0; y
< height
; y
++)
459 // optimization: draw identical adjacent pixels together.
460 for (int x
= 0; x
< width
; x
++)
462 COLORREF col
= ::GetPixel(dcSrc
, x
, y
);
463 HBRUSH brush
= ::CreateSolidBrush( col
);
465 rect
.left
= xdest
+ x
;
466 rect
.top
= ydest
+ y
;
467 while( (x
+ 1 < width
) &&
468 (::GetPixel(dcSrc
, x
+ 1, y
) == col
) )
472 rect
.right
= xdest
+ x
+ 1;
473 rect
.bottom
= rect
.top
+ 1;
474 ::FillRect((HDC
) m_hDC
, &rect
, brush
);
475 ::DeleteObject(brush
);
485 // wxUSE_PRINTING_ARCHITECTURE