]> git.saurik.com Git - wxWidgets.git/blame - src/msw/printwin.cpp
Added wxPGProperty::OnValidationFailure(); needed it and some other tweaks to allow...
[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:
373c713a
VS
468 PageFragmentDCImpl(wxMemoryDC *owner, wxDC *printer,
469 const wxPoint& offset,
470 const wxSize& fullSize)
25a3fca2 471 : wxMemoryDCImpl(owner, printer),
373c713a
VS
472 m_offset(offset),
473 m_fullSize(fullSize)
25a3fca2
VS
474 {
475 SetDeviceOrigin(0, 0);
476 }
477
478 virtual void SetDeviceOrigin(wxCoord x, wxCoord y)
479 {
373c713a 480 wxMemoryDCImpl::SetDeviceOrigin(x - m_offset.x, y - m_offset.y);
25a3fca2
VS
481 }
482
483 virtual void DoGetDeviceOrigin(wxCoord *x, wxCoord *y) const
484 {
485 wxMemoryDCImpl::DoGetDeviceOrigin(x, y);
373c713a
VS
486 if ( x ) *x += m_offset.x;
487 if ( x ) *y += m_offset.y;
25a3fca2
VS
488 }
489
490 virtual void DoGetSize(int *width, int *height) const
491 {
492 if ( width )
373c713a 493 *width = m_fullSize.x;
25a3fca2 494 if ( height )
373c713a 495 *height = m_fullSize.y;
25a3fca2
VS
496 }
497
498private:
373c713a
VS
499 wxPoint m_offset;
500 wxSize m_fullSize;
25a3fca2
VS
501};
502
503class PageFragmentDC : public wxDC
504{
505public:
373c713a
VS
506 PageFragmentDC(wxDC* printer, wxBitmap& bmp,
507 const wxPoint& offset,
508 const wxSize& fullSize)
509 : wxDC(new PageFragmentDCImpl((wxMemoryDC*)this, printer, offset, fullSize))
25a3fca2 510 {
5c33522f 511 static_cast<PageFragmentDCImpl*>(m_pimpl)->DoSelect(bmp);
25a3fca2
VS
512 }
513};
514
515// estimate how big chunks we can render, given available RAM
516long ComputeFragmentSize(long printerDepth,
517 long width,
518 long height)
519{
520 // Compute the amount of memory needed to generate the preview.
521 // Memory requirements of RenderPageFragment() are as follows:
522 //
523 // (memory DC - always)
524 // width * height * printerDepth/8
525 // (wxImage + wxDIB instance)
526 // width * height * (3 + 4)
527 // (this could be reduced to *3 if using wxGraphicsContext)
528 //
529 // So, given amount of memory M, we can render at most
530 //
531 // height = M / (width * (printerDepth/8 + F))
532 //
533 // where F is 3 or 7 depending on whether wxGraphicsContext is used or not
534
535 wxMemorySize memAvail = wxGetFreeMemory();
536 if ( memAvail == -1 )
537 {
538 // we don't know; 10meg shouldn't be a problem hopefully
539 memAvail = 10000000;
540 }
541 else
542 {
543 // limit ourselves to half of available RAM to have a margin for other
544 // apps, for our rendering code, and for miscalculations
545 memAvail /= 2;
546 }
547
548 const float perPixel = float(printerDepth)/8 + (3 + 4);
549
550 const long perLine = long(width * perPixel);
551 const long maxstep = (memAvail / perLine).GetValue();
552 const long step = wxMin(height, maxstep);
553
554 wxLogTrace("printing",
555 "using %liMB of RAM (%li lines) for preview, %li %lipx fragments",
556 long((memAvail >> 20).GetValue()),
557 maxstep,
558 (height+step-1) / step,
559 step);
560
561 return step;
562}
563
564} // anonymous namespace
565
566
567bool wxWindowsPrintPreview::RenderPageFragment(float scaleX, float scaleY,
568 int *nextFinalLine,
569 wxPrinterDC& printer,
570 wxMemoryDC& finalDC,
571 const wxRect& rect,
572 int pageNum)
573{
574 // compute 'rect' equivalent in the small final bitmap:
575 const wxRect smallRect(wxPoint(0, *nextFinalLine),
576 wxPoint(int(rect.GetRight() * scaleX),
577 int(rect.GetBottom() * scaleY)));
578 wxLogTrace("printing",
579 "rendering fragment of page %i: [%i,%i,%i,%i] scaled down to [%i,%i,%i,%i]",
580 pageNum,
581 rect.x, rect.y, rect.GetRight(), rect.GetBottom(),
582 smallRect.x, smallRect.y, smallRect.GetRight(), smallRect.GetBottom()
583 );
584
585 // create DC and bitmap compatible with printer DC:
586 wxBitmap large(rect.width, rect.height, printer);
587 if ( !large.IsOk() )
588 return false;
589
590 // render part of the page into it:
591 {
373c713a
VS
592 PageFragmentDC memoryDC(&printer, large,
593 rect.GetPosition(),
594 wxSize(m_pageWidth, m_pageHeight));
25a3fca2
VS
595 if ( !memoryDC.IsOk() )
596 return false;
597
598 memoryDC.Clear();
599
600 if ( !RenderPageIntoDC(memoryDC, pageNum) )
601 return false;
602 } // release bitmap from memoryDC
603
604 // now scale the rendered part down and blit it into final output:
605
606 wxImage img;
607 {
608 wxDIB dib(large);
609 if ( !dib.IsOk() )
610 return false;
611 large = wxNullBitmap; // free memory a.s.a.p.
612 img = dib.ConvertToImage();
613 } // free the DIB now that it's no longer needed, too
614
615 if ( !img.IsOk() )
616 return false;
617
618 img.Rescale(smallRect.width, smallRect.height, wxIMAGE_QUALITY_HIGH);
619 if ( !img.IsOk() )
620 return false;
621
622 wxBitmap bmp(img);
623 if ( !bmp.IsOk() )
624 return false;
625
626 img = wxNullImage;
627 finalDC.DrawBitmap(bmp, smallRect.x, smallRect.y);
628 if ( bmp.IsOk() )
629 {
630 *nextFinalLine += smallRect.height;
631 return true;
632 }
633
634 return false;
635}
636
637bool wxWindowsPrintPreview::RenderPageIntoBitmapHQ(wxBitmap& bmp, int pageNum)
638{
639 wxLogTrace("printing", "rendering HQ preview of page %i", pageNum);
640
641 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
642 if ( !printerDC.IsOk() )
643 return false;
644
645 // compute scale factor
646 const float scaleX = float(bmp.GetWidth()) / float(m_pageWidth);
647 const float scaleY = float(bmp.GetHeight()) / float(m_pageHeight);
648
649 wxMemoryDC bmpDC;
650 bmpDC.SelectObject(bmp);
651 bmpDC.Clear();
652
653 const int initialStep = ComputeFragmentSize(printerDC.GetDepth(),
654 m_pageWidth, m_pageHeight);
655
656 wxRect todo(0, 0, m_pageWidth, initialStep); // rect to render
657 int nextFinalLine = 0; // first not-yet-rendered output line
658
659 while ( todo.y < m_pageHeight )
660 {
661 todo.SetBottom(wxMin(todo.GetBottom(), m_pageHeight - 1));
662
663 if ( !RenderPageFragment(scaleX, scaleY,
664 &nextFinalLine,
665 printerDC,
666 bmpDC,
667 todo,
668 pageNum) )
669 {
670 if ( todo.height < 20 )
671 {
672 // something is very wrong if we can't render even at this
673 // slow space, let's bail out and fall back to low quality
674 // preview
675 wxLogTrace("printing",
676 "it seems that HQ preview doesn't work at all");
677 return false;
678 }
679
680 // it's possible our memory calculation was off, or conditions
681 // changed, or there's not enough _bitmap_ resources; try if using
682 // smaller bitmap would help:
683 todo.height /= 2;
684
685 wxLogTrace("printing",
686 "preview of fragment failed, reducing height to %ipx",
687 todo.height);
688
689 continue; // retry at the same position again
690 }
691
692 // move to the next segment
693 todo.Offset(0, todo.height);
694 }
695
696 return true;
697}
698
699bool wxWindowsPrintPreview::RenderPageIntoBitmap(wxBitmap& bmp, int pageNum)
700{
701 // try high quality rendering first:
702 if ( !m_hqPreviewFailed )
703 {
704 if ( RenderPageIntoBitmapHQ(bmp, pageNum) )
705 {
706 return true;
707 }
708 else
709 {
710 wxLogTrace("printing",
711 "high-quality preview failed, falling back to normal");
712 m_hqPreviewFailed = true; // don't bother re-trying
713 }
714 }
715
716 // if it fails, use the generic method that is less resource intensive,
717 // but inexact
718 return wxPrintPreviewBase::RenderPageIntoBitmap(bmp, pageNum);
719}
720
721#endif // wxUSE_HIGH_QUALITY_PREVIEW
722
2bda0e17
KB
723/****************************************************************************
724
7bcb11d3 725 FUNCTION: wxAbortProc()
103aec29 726
2bda0e17 727 PURPOSE: Processes messages for the Abort Dialog box
103aec29 728
2bda0e17
KB
729****************************************************************************/
730
731LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
732{
733 MSG msg;
103aec29 734
34da0970 735 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
2bda0e17 736 return(TRUE);
103aec29 737
2bda0e17 738 /* Process messages intended for the abort dialog box */
103aec29 739
078cf5cb 740 while (!wxPrinterBase::sm_abortIt && ::PeekMessage(&msg, 0, 0, 0, TRUE))
34da0970 741 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
2bda0e17
KB
742 TranslateMessage(&msg);
743 DispatchMessage(&msg);
744 }
103aec29 745
e71693c3 746 /* bAbort is TRUE (return is FALSE) if the user has aborted */
103aec29 747
e71693c3 748 return !wxPrinterBase::sm_abortIt;
2bda0e17
KB
749}
750
f6bcfd97
BP
751#endif
752 // wxUSE_PRINTING_ARCHITECTURE