]> git.saurik.com Git - wxWidgets.git/blame - src/msw/printwin.cpp
better guarding when no printing architecture exists (patch from Joel Low)
[wxWidgets.git] / src / msw / printwin.cpp
CommitLineData
2bda0e17 1/////////////////////////////////////////////////////////////////////////////
7520f3da 2// Name: src/msw/printwin.cpp
2bda0e17
KB
3// Purpose: wxWindowsPrinter framework
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
6c9a19aa 8// Copyright: (c) Julian Smart
65571936 9// Licence: wxWindows licence
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
103aec29
VZ
12// ===========================================================================
13// declarations
14// ===========================================================================
15
16// ---------------------------------------------------------------------------
17// headers
18// ---------------------------------------------------------------------------
19
2bda0e17
KB
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
103aec29 24 #pragma hdrstop
2bda0e17
KB
25#endif
26
12bdd77c
JS
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)
f6bcfd97 30
2bda0e17 31#ifndef WX_PRECOMP
57bd4c60 32 #include "wx/msw/wrapcdlg.h"
0c589ad0
BM
33 #include "wx/window.h"
34 #include "wx/msw/private.h"
103aec29
VZ
35 #include "wx/utils.h"
36 #include "wx/dc.h"
37 #include "wx/app.h"
38 #include "wx/msgdlg.h"
0c589ad0 39 #include "wx/intl.h"
e4db172a 40 #include "wx/log.h"
6d50343d 41 #include "wx/dcprint.h"
25a3fca2
VS
42 #include "wx/dcmemory.h"
43 #include "wx/image.h"
2bda0e17
KB
44#endif
45
25a3fca2
VS
46#include "wx/msw/dib.h"
47#include "wx/msw/dcmemory.h"
2bda0e17 48#include "wx/msw/printwin.h"
f4322df6 49#include "wx/msw/printdlg.h"
2bda0e17 50#include "wx/msw/private.h"
888dde65 51#include "wx/msw/dcprint.h"
2bda0e17
KB
52
53#include <stdlib.h>
2bda0e17 54
2bda0e17 55#ifndef __WIN32__
103aec29 56 #include <print.h>
2bda0e17
KB
57#endif
58
103aec29
VZ
59// ---------------------------------------------------------------------------
60// private functions
61// ---------------------------------------------------------------------------
62
2bda0e17
KB
63LONG APIENTRY _EXPORT wxAbortProc(HDC hPr, int Code);
64
103aec29
VZ
65// ---------------------------------------------------------------------------
66// wxWin macros
67// ---------------------------------------------------------------------------
68
103aec29
VZ
69 IMPLEMENT_DYNAMIC_CLASS(wxWindowsPrinter, wxPrinterBase)
70 IMPLEMENT_CLASS(wxWindowsPrintPreview, wxPrintPreviewBase)
2bda0e17 71
103aec29
VZ
72// ===========================================================================
73// implementation
74// ===========================================================================
75
76// ---------------------------------------------------------------------------
77// Printer
78// ---------------------------------------------------------------------------
7bcb11d3 79
103aec29
VZ
80wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData *data)
81 : wxPrinterBase(data)
2bda0e17 82{
d6f2a891 83 m_lpAbortProc = (WXFARPROC)wxAbortProc;
2bda0e17
KB
84}
85
103aec29 86wxWindowsPrinter::~wxWindowsPrinter()
2bda0e17 87{
33ac7e6f 88 // avoids warning about statement with no effect (FreeProcInstance
e41e579f 89 // doesn't do anything under Win32)
d66dcb60 90#if !defined(__WIN32__) && !defined(__NT__)
34da0970 91 FreeProcInstance((FARPROC) m_lpAbortProc);
a17e237f 92#endif
2bda0e17
KB
93}
94
95bool wxWindowsPrinter::Print(wxWindow *parent, wxPrintout *printout, bool prompt)
96{
e41e579f 97 sm_abortIt = false;
7bcb11d3 98 sm_abortWindow = NULL;
103aec29 99
7bcb11d3 100 if (!printout)
f6bcfd97
BP
101 {
102 sm_lastError = wxPRINTER_ERROR;
e41e579f 103 return false;
f6bcfd97 104 }
103aec29 105
e41e579f 106 printout->SetIsPreview(false);
1044a386 107
d2b354f9
JS
108 if (m_printDialogData.GetMinPage() < 1)
109 m_printDialogData.SetMinPage(1);
110 if (m_printDialogData.GetMaxPage() < 1)
111 m_printDialogData.SetMaxPage(9999);
2bda0e17 112
103aec29 113 // Create a suitable device context
f415cab9 114 wxPrinterDC *dc wxDUMMY_INITIALIZE(NULL);
7bcb11d3
JS
115 if (prompt)
116 {
f415cab9 117 dc = wxDynamicCast(PrintDialog(parent), wxPrinterDC);
7bcb11d3 118 if (!dc)
e41e579f 119 return false;
7bcb11d3
JS
120 }
121 else
122 {
7bcb11d3
JS
123 dc = new wxPrinterDC(m_printDialogData.GetPrintData());
124 }
103aec29 125
7bcb11d3 126 // May have pressed cancel.
888dde65 127 if (!dc || !dc->IsOk())
7bcb11d3
JS
128 {
129 if (dc) delete dc;
e41e579f 130 return false;
7bcb11d3 131 }
103aec29 132
888dde65
RR
133 wxPrinterDCImpl *impl = (wxPrinterDCImpl*) dc->GetImpl();
134
7bcb11d3 135 HDC hdc = ::GetDC(NULL);
5cb598ae
WS
136 int logPPIScreenX = ::GetDeviceCaps(hdc, LOGPIXELSX);
137 int logPPIScreenY = ::GetDeviceCaps(hdc, LOGPIXELSY);
7bcb11d3 138 ::ReleaseDC(NULL, hdc);
103aec29 139
888dde65
RR
140 int logPPIPrinterX = ::GetDeviceCaps((HDC) impl->GetHDC(), LOGPIXELSX);
141 int logPPIPrinterY = ::GetDeviceCaps((HDC) impl->GetHDC(), LOGPIXELSY);
7bcb11d3
JS
142 if (logPPIPrinterX == 0 || logPPIPrinterY == 0)
143 {
144 delete dc;
f6bcfd97 145 sm_lastError = wxPRINTER_ERROR;
e41e579f 146 return false;
7bcb11d3 147 }
103aec29 148
7bcb11d3
JS
149 printout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
150 printout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
103aec29
VZ
151
152 // Set printout parameters
7bcb11d3 153 printout->SetDC(dc);
103aec29 154
7bcb11d3
JS
155 int w, h;
156 dc->GetSize(&w, &h);
157 printout->SetPageSizePixels((int)w, (int)h);
f415cab9 158 printout->SetPaperRectPixels(dc->GetPaperRect());
d6a1743b 159
7bcb11d3
JS
160 dc->GetSizeMM(&w, &h);
161 printout->SetPageSizeMM((int)w, (int)h);
103aec29 162
7bcb11d3 163 // Create an abort window
0f07e3dc 164 wxBusyCursor busyCursor;
103aec29 165
1044a386
JS
166 printout->OnPreparePrinting();
167
d2b354f9
JS
168 // Get some parameters from the printout, if defined
169 int fromPage, toPage;
170 int minPage, maxPage;
171 printout->GetPageInfo(&minPage, &maxPage, &fromPage, &toPage);
172
173 if (maxPage == 0)
174 {
175 sm_lastError = wxPRINTER_ERROR;
e41e579f 176 return false;
d2b354f9
JS
177 }
178
179 // Only set min and max, because from and to have been
180 // set by the user
181 m_printDialogData.SetMinPage(minPage);
182 m_printDialogData.SetMaxPage(maxPage);
183
7bcb11d3
JS
184 wxWindow *win = CreateAbortWindow(parent, printout);
185 wxYield();
103aec29 186
f172cb82 187#if defined(__WATCOMC__) || defined(__BORLANDC__) || defined(__GNUWIN32__) || !defined(__WIN32__)
27a9bd48 188#ifdef STRICT
888dde65 189 ::SetAbortProc((HDC) impl->GetHDC(), (ABORTPROC) m_lpAbortProc);
27a9bd48 190#else
888dde65 191 ::SetAbortProc((HDC) impl->GetHDC(), (FARPROC) m_lpAbortProc);
27a9bd48 192#endif
7bcb11d3 193#else
888dde65 194 ::SetAbortProc((HDC) impl->GetHDC(), (int (_stdcall *)
7bcb11d3 195 // cast it to right type only if required
103aec29
VZ
196 // FIXME it's really cdecl and we're casting it to stdcall - either there is
197 // something I don't understand or it will crash at first usage
7bcb11d3
JS
198#ifdef STRICT
199 (HDC, int)
d6a1743b 200#else
7bcb11d3 201 ()
d6a1743b 202#endif
7bcb11d3
JS
203 )m_lpAbortProc);
204#endif
103aec29 205
7bcb11d3 206 if (!win)
2bda0e17 207 {
223d09f6 208 wxLogDebug(wxT("Could not create an abort dialog."));
f6bcfd97 209 sm_lastError = wxPRINTER_ERROR;
103aec29 210
7bcb11d3 211 delete dc;
e71693c3 212 return false;
2bda0e17 213 }
7bcb11d3 214 sm_abortWindow = win;
e41e579f 215 sm_abortWindow->Show();
103aec29
VZ
216 wxSafeYield();
217
7bcb11d3 218 printout->OnBeginPrinting();
103aec29 219
f6bcfd97
BP
220 sm_lastError = wxPRINTER_NO_ERROR;
221
e41e579f
DS
222 int minPageNum = minPage, maxPageNum = maxPage;
223
224 if ( !m_printDialogData.GetAllPages() )
e41e579f
DS
225 {
226 minPageNum = m_printDialogData.GetFromPage();
227 maxPageNum = m_printDialogData.GetToPage();
228 }
229
7bcb11d3 230 int copyCount;
70846f0a
VZ
231 for ( copyCount = 1;
232 copyCount <= m_printDialogData.GetNoCopies();
233 copyCount++ )
7bcb11d3 234 {
e41e579f 235 if ( !printout->OnBeginDocument(minPageNum, maxPageNum) )
7bcb11d3 236 {
103aec29 237 wxLogError(_("Could not start printing."));
f6bcfd97 238 sm_lastError = wxPRINTER_ERROR;
7bcb11d3
JS
239 break;
240 }
241 if (sm_abortIt)
f6bcfd97
BP
242 {
243 sm_lastError = wxPRINTER_CANCELLED;
7bcb11d3 244 break;
f6bcfd97 245 }
103aec29 246
7bcb11d3 247 int pn;
e41e579f
DS
248
249 for ( pn = minPageNum;
250 pn <= maxPageNum && printout->HasPage(pn);
70846f0a 251 pn++ )
7bcb11d3 252 {
70846f0a 253 if ( sm_abortIt )
7bcb11d3 254 {
f6bcfd97 255 sm_lastError = wxPRINTER_CANCELLED;
7bcb11d3
JS
256 break;
257 }
70846f0a
VZ
258
259 dc->StartPage();
260 bool cont = printout->OnPrintPage(pn);
261 dc->EndPage();
262
263 if ( !cont )
f6bcfd97
BP
264 {
265 sm_lastError = wxPRINTER_CANCELLED;
70846f0a 266 break;
f6bcfd97 267 }
7bcb11d3 268 }
70846f0a 269
7bcb11d3
JS
270 printout->OnEndDocument();
271 }
103aec29 272
7bcb11d3 273 printout->OnEndPrinting();
103aec29 274
7bcb11d3 275 if (sm_abortWindow)
2bda0e17 276 {
e41e579f 277 sm_abortWindow->Show(false);
7bcb11d3
JS
278 delete sm_abortWindow;
279 sm_abortWindow = NULL;
2bda0e17 280 }
103aec29 281
7bcb11d3 282 delete dc;
103aec29 283
e71693c3 284 return sm_lastError == wxPRINTER_NO_ERROR;
7bcb11d3 285}
2bda0e17 286
f415cab9 287wxDC *wxWindowsPrinter::PrintDialog(wxWindow *parent)
7bcb11d3 288{
f415cab9 289 wxDC *dc = (wxPrinterDC*) NULL;
2bda0e17 290
f415cab9 291 wxWindowsPrintDialog dialog(parent, & m_printDialogData);
7bcb11d3 292 int ret = dialog.ShowModal();
2bda0e17 293
7bcb11d3
JS
294 if (ret == wxID_OK)
295 {
296 dc = dialog.GetPrintDC();
297 m_printDialogData = dialog.GetPrintDialogData();
33ac7e6f 298 if (dc == NULL)
f6bcfd97
BP
299 sm_lastError = wxPRINTER_ERROR;
300 else
301 sm_lastError = wxPRINTER_NO_ERROR;
7bcb11d3 302 }
f6bcfd97
BP
303 else
304 sm_lastError = wxPRINTER_CANCELLED;
2bda0e17 305
7bcb11d3 306 return dc;
2bda0e17
KB
307}
308
a12f001e 309bool wxWindowsPrinter::Setup(wxWindow *WXUNUSED(parent))
2bda0e17 310{
b45dfd0a
RR
311#if 0
312 // We no longer expose that dialog
7bcb11d3 313 wxPrintDialog dialog(parent, & m_printDialogData);
e41e579f 314 dialog.GetPrintDialogData().SetSetupDialog(true);
7bcb11d3
JS
315
316 int ret = dialog.ShowModal();
317
318 if (ret == wxID_OK)
319 {
320 m_printDialogData = dialog.GetPrintDialogData();
321 }
322
323 return (ret == wxID_OK);
b45dfd0a 324#else
3950f9e1 325 return false;
b45dfd0a 326#endif
2bda0e17
KB
327}
328
329/*
7bcb11d3
JS
330* Print preview
331*/
2bda0e17 332
103aec29
VZ
333wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
334 wxPrintout *printoutForPrinting,
335 wxPrintDialogData *data)
336 : wxPrintPreviewBase(printout, printoutForPrinting, data)
2bda0e17 337{
25a3fca2
VS
338#if wxUSE_HIGH_QUALITY_PREVIEW
339 m_hqPreviewFailed = false;
340#endif
7bcb11d3 341 DetermineScaling();
2bda0e17
KB
342}
343
103aec29
VZ
344wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
345 wxPrintout *printoutForPrinting,
346 wxPrintData *data)
347 : wxPrintPreviewBase(printout, printoutForPrinting, data)
348{
25a3fca2
VS
349#if wxUSE_HIGH_QUALITY_PREVIEW
350 m_hqPreviewFailed = false;
351#endif
103aec29
VZ
352 DetermineScaling();
353}
354
355wxWindowsPrintPreview::~wxWindowsPrintPreview()
2bda0e17
KB
356{
357}
358
359bool wxWindowsPrintPreview::Print(bool interactive)
360{
7bcb11d3 361 if (!m_printPrintout)
e41e579f 362 return false;
7bcb11d3
JS
363 wxWindowsPrinter printer(&m_printDialogData);
364 return printer.Print(m_previewFrame, m_printPrintout, interactive);
2bda0e17
KB
365}
366
103aec29 367void wxWindowsPrintPreview::DetermineScaling()
2bda0e17 368{
f2a59080 369 ScreenHDC dc;
2bda0e17
KB
370 int logPPIScreenX = ::GetDeviceCaps(dc, LOGPIXELSX);
371 int logPPIScreenY = ::GetDeviceCaps(dc, LOGPIXELSY);
34da0970 372 m_previewPrintout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
103aec29 373
2bda0e17 374 // Get a device context for the currently selected printer
7bcb11d3 375 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
103aec29 376
f415cab9
JS
377 int printerWidthMM;
378 int printerHeightMM;
379 int printerXRes;
380 int printerYRes;
381 int logPPIPrinterX;
382 int logPPIPrinterY;
383
f4322df6 384 wxRect paperRect;
103aec29 385
888dde65 386 if ( printerDC.IsOk() )
2bda0e17 387 {
888dde65
RR
388 wxPrinterDCImpl *impl = (wxPrinterDCImpl*) printerDC.GetImpl();
389 HDC dc = GetHdcOf(*impl);
f415cab9
JS
390 printerWidthMM = ::GetDeviceCaps(dc, HORZSIZE);
391 printerHeightMM = ::GetDeviceCaps(dc, VERTSIZE);
f4fefc23
VZ
392 printerXRes = ::GetDeviceCaps(dc, HORZRES);
393 printerYRes = ::GetDeviceCaps(dc, VERTRES);
f415cab9
JS
394 logPPIPrinterX = ::GetDeviceCaps(dc, LOGPIXELSX);
395 logPPIPrinterY = ::GetDeviceCaps(dc, LOGPIXELSY);
103aec29 396
f4322df6 397 paperRect = printerDC.GetPaperRect();
103aec29 398
f2a59080
VZ
399 if ( logPPIPrinterX == 0 ||
400 logPPIPrinterY == 0 ||
f415cab9
JS
401 printerWidthMM == 0 ||
402 printerHeightMM == 0 )
f2a59080 403 {
e41e579f 404 m_isOk = false;
f2a59080 405 }
2bda0e17
KB
406 }
407 else
f2a59080 408 {
f415cab9
JS
409 // use some defaults
410 printerWidthMM = 150;
411 printerHeightMM = 250;
412 printerXRes = 1500;
413 printerYRes = 2500;
414 logPPIPrinterX = 600;
415 logPPIPrinterY = 600;
416
f4322df6 417 paperRect = wxRect(0, 0, printerXRes, printerYRes);
e41e579f 418 m_isOk = false;
f2a59080 419 }
34da0970
JS
420 m_pageWidth = printerXRes;
421 m_pageHeight = printerYRes;
f415cab9
JS
422 m_previewPrintout->SetPageSizePixels(printerXRes, printerYRes);
423 m_previewPrintout->SetPageSizeMM(printerWidthMM, printerHeightMM);
424 m_previewPrintout->SetPaperRectPixels(paperRect);
425 m_previewPrintout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
103aec29 426
2bda0e17 427 // At 100%, the page should look about page-size on the screen.
f415cab9
JS
428 m_previewScaleX = float(logPPIScreenX) / logPPIPrinterX;
429 m_previewScaleY = float(logPPIScreenY) / logPPIPrinterY;
2bda0e17
KB
430}
431
25a3fca2
VS
432
433#if wxUSE_HIGH_QUALITY_PREVIEW
434
435// The preview, as implemented in wxPrintPreviewBase (and as used prior to wx3)
436// is inexact: it uses screen DC, which has much lower resolution and has
437// other properties different from printer DC, so the preview is not quite
438// right.
439//
440// To make matters worse, if the application depends heavily on GetTextExtent()
441// or does text layout itself, the output in preview and on paper can be very
442// different. In particular, wxHtmlEasyPrinting is affected and the preview
443// can be easily off by several pages.
444//
445// To fix this, we attempt to render the preview into high-resolution bitmap
446// using DC with same resolution etc. as the printer DC. This takes lot of
447// memory, so the code is more complicated than it could be, but the results
448// are much better.
449//
450// Finally, this code is specific to wxMSW, because it doesn't make sense to
451// bother with it on other platforms. Both OSX and modern GNOME/GTK+
452// environments have builtin accurate preview (that applications should use
453// instead) and the differences between screen and printer DC in wxGTK are so
454// large than this trick doesn't help at all.
455
456namespace
457{
458
459// If there's not enough memory, we need to render the preview in parts.
460// Unfortunately we cannot simply use wxMemoryDC, because it reports its size
461// as bitmap's size, and we need to use smaller bitmap while still reporting
462// original ("correct") DC size, because printing code frequently uses
463// GetSize() to determine scaling factor. This DC class handles this.
464
465class PageFragmentDCImpl : public wxMemoryDCImpl
466{
467public:
468 PageFragmentDCImpl(wxMemoryDC *owner, wxDC *printer, const wxRect& rect)
469 : wxMemoryDCImpl(owner, printer),
470 m_rect(rect)
471 {
472 SetDeviceOrigin(0, 0);
473 }
474
475 virtual void SetDeviceOrigin(wxCoord x, wxCoord y)
476 {
477 wxMemoryDCImpl::SetDeviceOrigin(x - m_rect.x, y - m_rect.y);
478 }
479
480 virtual void DoGetDeviceOrigin(wxCoord *x, wxCoord *y) const
481 {
482 wxMemoryDCImpl::DoGetDeviceOrigin(x, y);
483 if ( x ) *x += m_rect.x;
484 if ( x ) *y += m_rect.y;
485 }
486
487 virtual void DoGetSize(int *width, int *height) const
488 {
489 if ( width )
490 *width = m_rect.width;
491 if ( height )
492 *height = m_rect.height;
493 }
494
495private:
496 wxRect m_rect;
497};
498
499class PageFragmentDC : public wxDC
500{
501public:
502 PageFragmentDC(wxDC* printer, wxBitmap& bmp, const wxRect& rect)
503 : wxDC(new PageFragmentDCImpl((wxMemoryDC*)this, printer, rect))
504 {
505 wx_static_cast(PageFragmentDCImpl*, m_pimpl)->DoSelect(bmp);
506 }
507};
508
509// estimate how big chunks we can render, given available RAM
510long ComputeFragmentSize(long printerDepth,
511 long width,
512 long height)
513{
514 // Compute the amount of memory needed to generate the preview.
515 // Memory requirements of RenderPageFragment() are as follows:
516 //
517 // (memory DC - always)
518 // width * height * printerDepth/8
519 // (wxImage + wxDIB instance)
520 // width * height * (3 + 4)
521 // (this could be reduced to *3 if using wxGraphicsContext)
522 //
523 // So, given amount of memory M, we can render at most
524 //
525 // height = M / (width * (printerDepth/8 + F))
526 //
527 // where F is 3 or 7 depending on whether wxGraphicsContext is used or not
528
529 wxMemorySize memAvail = wxGetFreeMemory();
530 if ( memAvail == -1 )
531 {
532 // we don't know; 10meg shouldn't be a problem hopefully
533 memAvail = 10000000;
534 }
535 else
536 {
537 // limit ourselves to half of available RAM to have a margin for other
538 // apps, for our rendering code, and for miscalculations
539 memAvail /= 2;
540 }
541
542 const float perPixel = float(printerDepth)/8 + (3 + 4);
543
544 const long perLine = long(width * perPixel);
545 const long maxstep = (memAvail / perLine).GetValue();
546 const long step = wxMin(height, maxstep);
547
548 wxLogTrace("printing",
549 "using %liMB of RAM (%li lines) for preview, %li %lipx fragments",
550 long((memAvail >> 20).GetValue()),
551 maxstep,
552 (height+step-1) / step,
553 step);
554
555 return step;
556}
557
558} // anonymous namespace
559
560
561bool wxWindowsPrintPreview::RenderPageFragment(float scaleX, float scaleY,
562 int *nextFinalLine,
563 wxPrinterDC& printer,
564 wxMemoryDC& finalDC,
565 const wxRect& rect,
566 int pageNum)
567{
568 // compute 'rect' equivalent in the small final bitmap:
569 const wxRect smallRect(wxPoint(0, *nextFinalLine),
570 wxPoint(int(rect.GetRight() * scaleX),
571 int(rect.GetBottom() * scaleY)));
572 wxLogTrace("printing",
573 "rendering fragment of page %i: [%i,%i,%i,%i] scaled down to [%i,%i,%i,%i]",
574 pageNum,
575 rect.x, rect.y, rect.GetRight(), rect.GetBottom(),
576 smallRect.x, smallRect.y, smallRect.GetRight(), smallRect.GetBottom()
577 );
578
579 // create DC and bitmap compatible with printer DC:
580 wxBitmap large(rect.width, rect.height, printer);
581 if ( !large.IsOk() )
582 return false;
583
584 // render part of the page into it:
585 {
586 PageFragmentDC memoryDC(&printer, large, rect);
587 if ( !memoryDC.IsOk() )
588 return false;
589
590 memoryDC.Clear();
591
592 if ( !RenderPageIntoDC(memoryDC, pageNum) )
593 return false;
594 } // release bitmap from memoryDC
595
596 // now scale the rendered part down and blit it into final output:
597
598 wxImage img;
599 {
600 wxDIB dib(large);
601 if ( !dib.IsOk() )
602 return false;
603 large = wxNullBitmap; // free memory a.s.a.p.
604 img = dib.ConvertToImage();
605 } // free the DIB now that it's no longer needed, too
606
607 if ( !img.IsOk() )
608 return false;
609
610 img.Rescale(smallRect.width, smallRect.height, wxIMAGE_QUALITY_HIGH);
611 if ( !img.IsOk() )
612 return false;
613
614 wxBitmap bmp(img);
615 if ( !bmp.IsOk() )
616 return false;
617
618 img = wxNullImage;
619 finalDC.DrawBitmap(bmp, smallRect.x, smallRect.y);
620 if ( bmp.IsOk() )
621 {
622 *nextFinalLine += smallRect.height;
623 return true;
624 }
625
626 return false;
627}
628
629bool wxWindowsPrintPreview::RenderPageIntoBitmapHQ(wxBitmap& bmp, int pageNum)
630{
631 wxLogTrace("printing", "rendering HQ preview of page %i", pageNum);
632
633 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
634 if ( !printerDC.IsOk() )
635 return false;
636
637 // compute scale factor
638 const float scaleX = float(bmp.GetWidth()) / float(m_pageWidth);
639 const float scaleY = float(bmp.GetHeight()) / float(m_pageHeight);
640
641 wxMemoryDC bmpDC;
642 bmpDC.SelectObject(bmp);
643 bmpDC.Clear();
644
645 const int initialStep = ComputeFragmentSize(printerDC.GetDepth(),
646 m_pageWidth, m_pageHeight);
647
648 wxRect todo(0, 0, m_pageWidth, initialStep); // rect to render
649 int nextFinalLine = 0; // first not-yet-rendered output line
650
651 while ( todo.y < m_pageHeight )
652 {
653 todo.SetBottom(wxMin(todo.GetBottom(), m_pageHeight - 1));
654
655 if ( !RenderPageFragment(scaleX, scaleY,
656 &nextFinalLine,
657 printerDC,
658 bmpDC,
659 todo,
660 pageNum) )
661 {
662 if ( todo.height < 20 )
663 {
664 // something is very wrong if we can't render even at this
665 // slow space, let's bail out and fall back to low quality
666 // preview
667 wxLogTrace("printing",
668 "it seems that HQ preview doesn't work at all");
669 return false;
670 }
671
672 // it's possible our memory calculation was off, or conditions
673 // changed, or there's not enough _bitmap_ resources; try if using
674 // smaller bitmap would help:
675 todo.height /= 2;
676
677 wxLogTrace("printing",
678 "preview of fragment failed, reducing height to %ipx",
679 todo.height);
680
681 continue; // retry at the same position again
682 }
683
684 // move to the next segment
685 todo.Offset(0, todo.height);
686 }
687
688 return true;
689}
690
691bool wxWindowsPrintPreview::RenderPageIntoBitmap(wxBitmap& bmp, int pageNum)
692{
693 // try high quality rendering first:
694 if ( !m_hqPreviewFailed )
695 {
696 if ( RenderPageIntoBitmapHQ(bmp, pageNum) )
697 {
698 return true;
699 }
700 else
701 {
702 wxLogTrace("printing",
703 "high-quality preview failed, falling back to normal");
704 m_hqPreviewFailed = true; // don't bother re-trying
705 }
706 }
707
708 // if it fails, use the generic method that is less resource intensive,
709 // but inexact
710 return wxPrintPreviewBase::RenderPageIntoBitmap(bmp, pageNum);
711}
712
713#endif // wxUSE_HIGH_QUALITY_PREVIEW
714
2bda0e17
KB
715/****************************************************************************
716
7bcb11d3 717 FUNCTION: wxAbortProc()
103aec29 718
2bda0e17 719 PURPOSE: Processes messages for the Abort Dialog box
103aec29 720
2bda0e17
KB
721****************************************************************************/
722
723LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
724{
725 MSG msg;
103aec29 726
34da0970 727 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
2bda0e17 728 return(TRUE);
103aec29 729
2bda0e17 730 /* Process messages intended for the abort dialog box */
103aec29 731
078cf5cb 732 while (!wxPrinterBase::sm_abortIt && ::PeekMessage(&msg, 0, 0, 0, TRUE))
34da0970 733 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
2bda0e17
KB
734 TranslateMessage(&msg);
735 DispatchMessage(&msg);
736 }
103aec29 737
e71693c3 738 /* bAbort is TRUE (return is FALSE) if the user has aborted */
103aec29 739
e71693c3 740 return !wxPrinterBase::sm_abortIt;
2bda0e17
KB
741}
742
f6bcfd97
BP
743#endif
744 // wxUSE_PRINTING_ARCHITECTURE