]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/printwin.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/printwin.cpp
3 // Purpose: wxWindowsPrinter framework
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 // Don't use the Windows printer if we're in wxUniv mode and using
28 // the PostScript architecture
29 #if wxUSE_PRINTING_ARCHITECTURE && (!defined(__WXUNIVERSAL__) || !wxUSE_POSTSCRIPT_ARCHITECTURE_IN_MSW)
32 #include "wx/msw/wrapcdlg.h"
33 #include "wx/window.h"
34 #include "wx/msw/private.h"
38 #include "wx/msgdlg.h"
41 #include "wx/dcprint.h"
42 #include "wx/dcmemory.h"
46 #include "wx/msw/dib.h"
47 #include "wx/msw/dcmemory.h"
48 #include "wx/msw/printwin.h"
49 #include "wx/msw/printdlg.h"
50 #include "wx/msw/private.h"
51 #include "wx/msw/dcprint.h"
52 #include "wx/msw/enhmeta.h"
60 // ---------------------------------------------------------------------------
62 // ---------------------------------------------------------------------------
64 LONG APIENTRY _EXPORT
wxAbortProc(HDC hPr
, int Code
);
66 // ---------------------------------------------------------------------------
68 // ---------------------------------------------------------------------------
70 IMPLEMENT_DYNAMIC_CLASS(wxWindowsPrinter
, wxPrinterBase
)
71 IMPLEMENT_CLASS(wxWindowsPrintPreview
, wxPrintPreviewBase
)
73 // ===========================================================================
75 // ===========================================================================
77 // ---------------------------------------------------------------------------
79 // ---------------------------------------------------------------------------
81 wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData
*data
)
84 m_lpAbortProc
= (WXFARPROC
)wxAbortProc
;
87 wxWindowsPrinter::~wxWindowsPrinter()
89 // avoids warning about statement with no effect (FreeProcInstance
90 // doesn't do anything under Win32)
91 #if !defined(__WIN32__) && !defined(__NT__)
92 FreeProcInstance((FARPROC
) m_lpAbortProc
);
96 bool wxWindowsPrinter::Print(wxWindow
*parent
, wxPrintout
*printout
, bool prompt
)
99 sm_abortWindow
= NULL
;
103 sm_lastError
= wxPRINTER_ERROR
;
107 printout
->SetIsPreview(false);
109 if (m_printDialogData
.GetMinPage() < 1)
110 m_printDialogData
.SetMinPage(1);
111 if (m_printDialogData
.GetMaxPage() < 1)
112 m_printDialogData
.SetMaxPage(9999);
114 // Create a suitable device context
115 wxPrinterDC
*dc
wxDUMMY_INITIALIZE(NULL
);
118 dc
= wxDynamicCast(PrintDialog(parent
), wxPrinterDC
);
124 dc
= new wxPrinterDC(m_printDialogData
.GetPrintData());
127 // May have pressed cancel.
128 if (!dc
|| !dc
->IsOk())
134 wxPrinterDCImpl
*impl
= (wxPrinterDCImpl
*) dc
->GetImpl();
136 HDC hdc
= ::GetDC(NULL
);
137 int logPPIScreenX
= ::GetDeviceCaps(hdc
, LOGPIXELSX
);
138 int logPPIScreenY
= ::GetDeviceCaps(hdc
, LOGPIXELSY
);
139 ::ReleaseDC(NULL
, hdc
);
141 int logPPIPrinterX
= ::GetDeviceCaps((HDC
) impl
->GetHDC(), LOGPIXELSX
);
142 int logPPIPrinterY
= ::GetDeviceCaps((HDC
) impl
->GetHDC(), LOGPIXELSY
);
143 if (logPPIPrinterX
== 0 || logPPIPrinterY
== 0)
146 sm_lastError
= wxPRINTER_ERROR
;
150 printout
->SetPPIScreen(logPPIScreenX
, logPPIScreenY
);
151 printout
->SetPPIPrinter(logPPIPrinterX
, logPPIPrinterY
);
153 // Set printout parameters
158 printout
->SetPageSizePixels((int)w
, (int)h
);
159 printout
->SetPaperRectPixels(dc
->GetPaperRect());
161 dc
->GetSizeMM(&w
, &h
);
162 printout
->SetPageSizeMM((int)w
, (int)h
);
164 // Create an abort window
165 wxBusyCursor busyCursor
;
167 printout
->OnPreparePrinting();
169 // Get some parameters from the printout, if defined
170 int fromPage
, toPage
;
171 int minPage
, maxPage
;
172 printout
->GetPageInfo(&minPage
, &maxPage
, &fromPage
, &toPage
);
176 sm_lastError
= wxPRINTER_ERROR
;
180 // Only set min and max, because from and to have been
182 m_printDialogData
.SetMinPage(minPage
);
183 m_printDialogData
.SetMaxPage(maxPage
);
185 wxWindow
*win
= CreateAbortWindow(parent
, printout
);
188 #if defined(__WATCOMC__) || defined(__BORLANDC__) || defined(__GNUWIN32__) || !defined(__WIN32__)
190 ::SetAbortProc((HDC
) impl
->GetHDC(), (ABORTPROC
) m_lpAbortProc
);
192 ::SetAbortProc((HDC
) impl
->GetHDC(), (FARPROC
) m_lpAbortProc
);
195 ::SetAbortProc((HDC
) impl
->GetHDC(), (int (_stdcall
*)
196 // cast it to right type only if required
197 // FIXME it's really cdecl and we're casting it to stdcall - either there is
198 // something I don't understand or it will crash at first usage
209 wxLogDebug(wxT("Could not create an abort dialog."));
210 sm_lastError
= wxPRINTER_ERROR
;
215 sm_abortWindow
= win
;
216 sm_abortWindow
->Show();
219 printout
->OnBeginPrinting();
221 sm_lastError
= wxPRINTER_NO_ERROR
;
223 int minPageNum
= minPage
, maxPageNum
= maxPage
;
225 if ( !m_printDialogData
.GetAllPages() )
227 minPageNum
= m_printDialogData
.GetFromPage();
228 maxPageNum
= m_printDialogData
.GetToPage();
233 copyCount
<= m_printDialogData
.GetNoCopies();
236 if ( !printout
->OnBeginDocument(minPageNum
, maxPageNum
) )
238 wxLogError(_("Could not start printing."));
239 sm_lastError
= wxPRINTER_ERROR
;
244 sm_lastError
= wxPRINTER_CANCELLED
;
250 for ( pn
= minPageNum
;
251 pn
<= maxPageNum
&& printout
->HasPage(pn
);
256 sm_lastError
= wxPRINTER_CANCELLED
;
261 bool cont
= printout
->OnPrintPage(pn
);
266 sm_lastError
= wxPRINTER_CANCELLED
;
271 printout
->OnEndDocument();
274 printout
->OnEndPrinting();
278 sm_abortWindow
->Show(false);
279 delete sm_abortWindow
;
280 sm_abortWindow
= NULL
;
285 return sm_lastError
== wxPRINTER_NO_ERROR
;
288 wxDC
*wxWindowsPrinter::PrintDialog(wxWindow
*parent
)
292 wxWindowsPrintDialog
dialog(parent
, & m_printDialogData
);
293 int ret
= dialog
.ShowModal();
297 dc
= dialog
.GetPrintDC();
298 m_printDialogData
= dialog
.GetPrintDialogData();
300 sm_lastError
= wxPRINTER_ERROR
;
302 sm_lastError
= wxPRINTER_NO_ERROR
;
305 sm_lastError
= wxPRINTER_CANCELLED
;
310 bool wxWindowsPrinter::Setup(wxWindow
*WXUNUSED(parent
))
313 // We no longer expose that dialog
314 wxPrintDialog
dialog(parent
, & m_printDialogData
);
315 dialog
.GetPrintDialogData().SetSetupDialog(true);
317 int ret
= dialog
.ShowModal();
321 m_printDialogData
= dialog
.GetPrintDialogData();
324 return (ret
== wxID_OK
);
334 wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout
*printout
,
335 wxPrintout
*printoutForPrinting
,
336 wxPrintDialogData
*data
)
337 : wxPrintPreviewBase(printout
, printoutForPrinting
, data
)
342 wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout
*printout
,
343 wxPrintout
*printoutForPrinting
,
345 : wxPrintPreviewBase(printout
, printoutForPrinting
, data
)
350 wxWindowsPrintPreview::~wxWindowsPrintPreview()
354 bool wxWindowsPrintPreview::Print(bool interactive
)
356 if (!m_printPrintout
)
358 wxWindowsPrinter
printer(&m_printDialogData
);
359 return printer
.Print(m_previewFrame
, m_printPrintout
, interactive
);
362 void wxWindowsPrintPreview::DetermineScaling()
365 int logPPIScreenX
= ::GetDeviceCaps(dc
, LOGPIXELSX
);
366 int logPPIScreenY
= ::GetDeviceCaps(dc
, LOGPIXELSY
);
367 m_previewPrintout
->SetPPIScreen(logPPIScreenX
, logPPIScreenY
);
369 // Get a device context for the currently selected printer
370 wxPrinterDC
printerDC(m_printDialogData
.GetPrintData());
381 if ( printerDC
.IsOk() )
383 wxPrinterDCImpl
*impl
= (wxPrinterDCImpl
*) printerDC
.GetImpl();
384 HDC dc
= GetHdcOf(*impl
);
385 printerWidthMM
= ::GetDeviceCaps(dc
, HORZSIZE
);
386 printerHeightMM
= ::GetDeviceCaps(dc
, VERTSIZE
);
387 printerXRes
= ::GetDeviceCaps(dc
, HORZRES
);
388 printerYRes
= ::GetDeviceCaps(dc
, VERTRES
);
389 logPPIPrinterX
= ::GetDeviceCaps(dc
, LOGPIXELSX
);
390 logPPIPrinterY
= ::GetDeviceCaps(dc
, LOGPIXELSY
);
392 paperRect
= printerDC
.GetPaperRect();
394 if ( logPPIPrinterX
== 0 ||
395 logPPIPrinterY
== 0 ||
396 printerWidthMM
== 0 ||
397 printerHeightMM
== 0 )
405 printerWidthMM
= 150;
406 printerHeightMM
= 250;
409 logPPIPrinterX
= 600;
410 logPPIPrinterY
= 600;
412 paperRect
= wxRect(0, 0, printerXRes
, printerYRes
);
415 m_pageWidth
= printerXRes
;
416 m_pageHeight
= printerYRes
;
417 m_previewPrintout
->SetPageSizePixels(printerXRes
, printerYRes
);
418 m_previewPrintout
->SetPageSizeMM(printerWidthMM
, printerHeightMM
);
419 m_previewPrintout
->SetPaperRectPixels(paperRect
);
420 m_previewPrintout
->SetPPIPrinter(logPPIPrinterX
, logPPIPrinterY
);
422 // At 100%, the page should look about page-size on the screen.
423 m_previewScaleX
= float(logPPIScreenX
) / logPPIPrinterX
;
424 m_previewScaleY
= float(logPPIScreenY
) / logPPIPrinterY
;
427 bool wxWindowsPrintPreview::RenderPageIntoBitmap(wxBitmap
& bmp
, int pageNum
)
429 // The preview, as implemented in wxPrintPreviewBase (and as used prior to
430 // wx3) is inexact: it uses screen DC, which has much lower resolution and
431 // has other properties different from printer DC, so the preview is not
434 // To make matters worse, if the application depends heavily on
435 // GetTextExtent() or does text layout itself, the output in preview and on
436 // paper can be very different. In particular, wxHtmlEasyPrinting is
437 // affected and the preview can be easily off by several pages.
439 // To fix this, we render the preview into high-resolution enhanced
440 // metafile with properties identical to the printer DC. This guarantees
441 // metrics correctness while still being fast.
444 // print the preview into a metafile:
445 wxPrinterDC
printerDC(m_printDialogData
.GetPrintData());
446 wxEnhMetaFileDC
metaDC(printerDC
,
448 printerDC
.GetSize().x
, printerDC
.GetSize().y
);
450 if ( !RenderPageIntoDC(metaDC
, pageNum
) )
453 wxEnhMetaFile
*metafile
= metaDC
.Close();
457 // now render the metafile:
459 bmpDC
.SelectObject(bmp
);
462 wxRect
outRect(0, 0, bmp
.GetWidth(), bmp
.GetHeight());
463 metafile
->Play(&bmpDC
, &outRect
);
468 // TODO: we should keep the metafile and reuse it when changing zoom level
474 /****************************************************************************
476 FUNCTION: wxAbortProc()
478 PURPOSE: Processes messages for the Abort Dialog box
480 ****************************************************************************/
482 LONG APIENTRY _EXPORT
wxAbortProc(HDC
WXUNUSED(hPr
), int WXUNUSED(Code
))
486 if (!wxPrinterBase::sm_abortWindow
) /* If the abort dialog isn't up yet */
489 /* Process messages intended for the abort dialog box */
491 while (!wxPrinterBase::sm_abortIt
&& ::PeekMessage(&msg
, 0, 0, 0, TRUE
))
492 if (!IsDialogMessage((HWND
) wxPrinterBase::sm_abortWindow
->GetHWND(), &msg
)) {
493 TranslateMessage(&msg
);
494 DispatchMessage(&msg
);
497 /* bAbort is TRUE (return is FALSE) if the user has aborted */
499 return !wxPrinterBase::sm_abortIt
;
503 // wxUSE_PRINTING_ARCHITECTURE