]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/printwin.cpp
better guarding when no printing architecture exists (patch from Joel Low)
[wxWidgets.git] / src / msw / printwin.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/printwin.cpp
3// Purpose: wxWindowsPrinter framework
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// ===========================================================================
13// declarations
14// ===========================================================================
15
16// ---------------------------------------------------------------------------
17// headers
18// ---------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
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)
30
31#ifndef WX_PRECOMP
32 #include "wx/msw/wrapcdlg.h"
33 #include "wx/window.h"
34 #include "wx/msw/private.h"
35 #include "wx/utils.h"
36 #include "wx/dc.h"
37 #include "wx/app.h"
38 #include "wx/msgdlg.h"
39 #include "wx/intl.h"
40 #include "wx/log.h"
41 #include "wx/dcprint.h"
42 #include "wx/dcmemory.h"
43 #include "wx/image.h"
44#endif
45
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
53#include <stdlib.h>
54
55#ifndef __WIN32__
56 #include <print.h>
57#endif
58
59// ---------------------------------------------------------------------------
60// private functions
61// ---------------------------------------------------------------------------
62
63LONG APIENTRY _EXPORT wxAbortProc(HDC hPr, int Code);
64
65// ---------------------------------------------------------------------------
66// wxWin macros
67// ---------------------------------------------------------------------------
68
69 IMPLEMENT_DYNAMIC_CLASS(wxWindowsPrinter, wxPrinterBase)
70 IMPLEMENT_CLASS(wxWindowsPrintPreview, wxPrintPreviewBase)
71
72// ===========================================================================
73// implementation
74// ===========================================================================
75
76// ---------------------------------------------------------------------------
77// Printer
78// ---------------------------------------------------------------------------
79
80wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData *data)
81 : wxPrinterBase(data)
82{
83 m_lpAbortProc = (WXFARPROC)wxAbortProc;
84}
85
86wxWindowsPrinter::~wxWindowsPrinter()
87{
88 // avoids warning about statement with no effect (FreeProcInstance
89 // doesn't do anything under Win32)
90#if !defined(__WIN32__) && !defined(__NT__)
91 FreeProcInstance((FARPROC) m_lpAbortProc);
92#endif
93}
94
95bool wxWindowsPrinter::Print(wxWindow *parent, wxPrintout *printout, bool prompt)
96{
97 sm_abortIt = false;
98 sm_abortWindow = NULL;
99
100 if (!printout)
101 {
102 sm_lastError = wxPRINTER_ERROR;
103 return false;
104 }
105
106 printout->SetIsPreview(false);
107
108 if (m_printDialogData.GetMinPage() < 1)
109 m_printDialogData.SetMinPage(1);
110 if (m_printDialogData.GetMaxPage() < 1)
111 m_printDialogData.SetMaxPage(9999);
112
113 // Create a suitable device context
114 wxPrinterDC *dc wxDUMMY_INITIALIZE(NULL);
115 if (prompt)
116 {
117 dc = wxDynamicCast(PrintDialog(parent), wxPrinterDC);
118 if (!dc)
119 return false;
120 }
121 else
122 {
123 dc = new wxPrinterDC(m_printDialogData.GetPrintData());
124 }
125
126 // May have pressed cancel.
127 if (!dc || !dc->IsOk())
128 {
129 if (dc) delete dc;
130 return false;
131 }
132
133 wxPrinterDCImpl *impl = (wxPrinterDCImpl*) dc->GetImpl();
134
135 HDC hdc = ::GetDC(NULL);
136 int logPPIScreenX = ::GetDeviceCaps(hdc, LOGPIXELSX);
137 int logPPIScreenY = ::GetDeviceCaps(hdc, LOGPIXELSY);
138 ::ReleaseDC(NULL, hdc);
139
140 int logPPIPrinterX = ::GetDeviceCaps((HDC) impl->GetHDC(), LOGPIXELSX);
141 int logPPIPrinterY = ::GetDeviceCaps((HDC) impl->GetHDC(), LOGPIXELSY);
142 if (logPPIPrinterX == 0 || logPPIPrinterY == 0)
143 {
144 delete dc;
145 sm_lastError = wxPRINTER_ERROR;
146 return false;
147 }
148
149 printout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
150 printout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
151
152 // Set printout parameters
153 printout->SetDC(dc);
154
155 int w, h;
156 dc->GetSize(&w, &h);
157 printout->SetPageSizePixels((int)w, (int)h);
158 printout->SetPaperRectPixels(dc->GetPaperRect());
159
160 dc->GetSizeMM(&w, &h);
161 printout->SetPageSizeMM((int)w, (int)h);
162
163 // Create an abort window
164 wxBusyCursor busyCursor;
165
166 printout->OnPreparePrinting();
167
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;
176 return false;
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
184 wxWindow *win = CreateAbortWindow(parent, printout);
185 wxYield();
186
187#if defined(__WATCOMC__) || defined(__BORLANDC__) || defined(__GNUWIN32__) || !defined(__WIN32__)
188#ifdef STRICT
189 ::SetAbortProc((HDC) impl->GetHDC(), (ABORTPROC) m_lpAbortProc);
190#else
191 ::SetAbortProc((HDC) impl->GetHDC(), (FARPROC) m_lpAbortProc);
192#endif
193#else
194 ::SetAbortProc((HDC) impl->GetHDC(), (int (_stdcall *)
195 // cast it to right type only if required
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
198#ifdef STRICT
199 (HDC, int)
200#else
201 ()
202#endif
203 )m_lpAbortProc);
204#endif
205
206 if (!win)
207 {
208 wxLogDebug(wxT("Could not create an abort dialog."));
209 sm_lastError = wxPRINTER_ERROR;
210
211 delete dc;
212 return false;
213 }
214 sm_abortWindow = win;
215 sm_abortWindow->Show();
216 wxSafeYield();
217
218 printout->OnBeginPrinting();
219
220 sm_lastError = wxPRINTER_NO_ERROR;
221
222 int minPageNum = minPage, maxPageNum = maxPage;
223
224 if ( !m_printDialogData.GetAllPages() )
225 {
226 minPageNum = m_printDialogData.GetFromPage();
227 maxPageNum = m_printDialogData.GetToPage();
228 }
229
230 int copyCount;
231 for ( copyCount = 1;
232 copyCount <= m_printDialogData.GetNoCopies();
233 copyCount++ )
234 {
235 if ( !printout->OnBeginDocument(minPageNum, maxPageNum) )
236 {
237 wxLogError(_("Could not start printing."));
238 sm_lastError = wxPRINTER_ERROR;
239 break;
240 }
241 if (sm_abortIt)
242 {
243 sm_lastError = wxPRINTER_CANCELLED;
244 break;
245 }
246
247 int pn;
248
249 for ( pn = minPageNum;
250 pn <= maxPageNum && printout->HasPage(pn);
251 pn++ )
252 {
253 if ( sm_abortIt )
254 {
255 sm_lastError = wxPRINTER_CANCELLED;
256 break;
257 }
258
259 dc->StartPage();
260 bool cont = printout->OnPrintPage(pn);
261 dc->EndPage();
262
263 if ( !cont )
264 {
265 sm_lastError = wxPRINTER_CANCELLED;
266 break;
267 }
268 }
269
270 printout->OnEndDocument();
271 }
272
273 printout->OnEndPrinting();
274
275 if (sm_abortWindow)
276 {
277 sm_abortWindow->Show(false);
278 delete sm_abortWindow;
279 sm_abortWindow = NULL;
280 }
281
282 delete dc;
283
284 return sm_lastError == wxPRINTER_NO_ERROR;
285}
286
287wxDC *wxWindowsPrinter::PrintDialog(wxWindow *parent)
288{
289 wxDC *dc = (wxPrinterDC*) NULL;
290
291 wxWindowsPrintDialog dialog(parent, & m_printDialogData);
292 int ret = dialog.ShowModal();
293
294 if (ret == wxID_OK)
295 {
296 dc = dialog.GetPrintDC();
297 m_printDialogData = dialog.GetPrintDialogData();
298 if (dc == NULL)
299 sm_lastError = wxPRINTER_ERROR;
300 else
301 sm_lastError = wxPRINTER_NO_ERROR;
302 }
303 else
304 sm_lastError = wxPRINTER_CANCELLED;
305
306 return dc;
307}
308
309bool wxWindowsPrinter::Setup(wxWindow *WXUNUSED(parent))
310{
311#if 0
312 // We no longer expose that dialog
313 wxPrintDialog dialog(parent, & m_printDialogData);
314 dialog.GetPrintDialogData().SetSetupDialog(true);
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);
324#else
325 return false;
326#endif
327}
328
329/*
330* Print preview
331*/
332
333wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
334 wxPrintout *printoutForPrinting,
335 wxPrintDialogData *data)
336 : wxPrintPreviewBase(printout, printoutForPrinting, data)
337{
338#if wxUSE_HIGH_QUALITY_PREVIEW
339 m_hqPreviewFailed = false;
340#endif
341 DetermineScaling();
342}
343
344wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
345 wxPrintout *printoutForPrinting,
346 wxPrintData *data)
347 : wxPrintPreviewBase(printout, printoutForPrinting, data)
348{
349#if wxUSE_HIGH_QUALITY_PREVIEW
350 m_hqPreviewFailed = false;
351#endif
352 DetermineScaling();
353}
354
355wxWindowsPrintPreview::~wxWindowsPrintPreview()
356{
357}
358
359bool wxWindowsPrintPreview::Print(bool interactive)
360{
361 if (!m_printPrintout)
362 return false;
363 wxWindowsPrinter printer(&m_printDialogData);
364 return printer.Print(m_previewFrame, m_printPrintout, interactive);
365}
366
367void wxWindowsPrintPreview::DetermineScaling()
368{
369 ScreenHDC dc;
370 int logPPIScreenX = ::GetDeviceCaps(dc, LOGPIXELSX);
371 int logPPIScreenY = ::GetDeviceCaps(dc, LOGPIXELSY);
372 m_previewPrintout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
373
374 // Get a device context for the currently selected printer
375 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
376
377 int printerWidthMM;
378 int printerHeightMM;
379 int printerXRes;
380 int printerYRes;
381 int logPPIPrinterX;
382 int logPPIPrinterY;
383
384 wxRect paperRect;
385
386 if ( printerDC.IsOk() )
387 {
388 wxPrinterDCImpl *impl = (wxPrinterDCImpl*) printerDC.GetImpl();
389 HDC dc = GetHdcOf(*impl);
390 printerWidthMM = ::GetDeviceCaps(dc, HORZSIZE);
391 printerHeightMM = ::GetDeviceCaps(dc, VERTSIZE);
392 printerXRes = ::GetDeviceCaps(dc, HORZRES);
393 printerYRes = ::GetDeviceCaps(dc, VERTRES);
394 logPPIPrinterX = ::GetDeviceCaps(dc, LOGPIXELSX);
395 logPPIPrinterY = ::GetDeviceCaps(dc, LOGPIXELSY);
396
397 paperRect = printerDC.GetPaperRect();
398
399 if ( logPPIPrinterX == 0 ||
400 logPPIPrinterY == 0 ||
401 printerWidthMM == 0 ||
402 printerHeightMM == 0 )
403 {
404 m_isOk = false;
405 }
406 }
407 else
408 {
409 // use some defaults
410 printerWidthMM = 150;
411 printerHeightMM = 250;
412 printerXRes = 1500;
413 printerYRes = 2500;
414 logPPIPrinterX = 600;
415 logPPIPrinterY = 600;
416
417 paperRect = wxRect(0, 0, printerXRes, printerYRes);
418 m_isOk = false;
419 }
420 m_pageWidth = printerXRes;
421 m_pageHeight = printerYRes;
422 m_previewPrintout->SetPageSizePixels(printerXRes, printerYRes);
423 m_previewPrintout->SetPageSizeMM(printerWidthMM, printerHeightMM);
424 m_previewPrintout->SetPaperRectPixels(paperRect);
425 m_previewPrintout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
426
427 // At 100%, the page should look about page-size on the screen.
428 m_previewScaleX = float(logPPIScreenX) / logPPIPrinterX;
429 m_previewScaleY = float(logPPIScreenY) / logPPIPrinterY;
430}
431
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
715/****************************************************************************
716
717 FUNCTION: wxAbortProc()
718
719 PURPOSE: Processes messages for the Abort Dialog box
720
721****************************************************************************/
722
723LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
724{
725 MSG msg;
726
727 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
728 return(TRUE);
729
730 /* Process messages intended for the abort dialog box */
731
732 while (!wxPrinterBase::sm_abortIt && ::PeekMessage(&msg, 0, 0, 0, TRUE))
733 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
734 TranslateMessage(&msg);
735 DispatchMessage(&msg);
736 }
737
738 /* bAbort is TRUE (return is FALSE) if the user has aborted */
739
740 return !wxPrinterBase::sm_abortIt;
741}
742
743#endif
744 // wxUSE_PRINTING_ARCHITECTURE