]> git.saurik.com Git - wxWidgets.git/blob - src/msw/printwin.cpp
wxHashSet::count() method should be const
[wxWidgets.git] / src / msw / printwin.cpp
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
63 LONG 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
80 wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData *data)
81 : wxPrinterBase(data)
82 {
83 m_lpAbortProc = (WXFARPROC)wxAbortProc;
84 }
85
86 wxWindowsPrinter::~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
95 bool 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
287 wxDC *wxWindowsPrinter::PrintDialog(wxWindow *parent)
288 {
289 wxDC *dc = 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
309 bool 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
333 wxWindowsPrintPreview::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
344 wxWindowsPrintPreview::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
355 wxWindowsPrintPreview::~wxWindowsPrintPreview()
356 {
357 }
358
359 bool 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
367 void 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
456 namespace
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
465 class PageFragmentDCImpl : public wxMemoryDCImpl
466 {
467 public:
468 PageFragmentDCImpl(wxMemoryDC *owner, wxDC *printer,
469 const wxPoint& offset,
470 const wxSize& fullSize)
471 : wxMemoryDCImpl(owner, printer),
472 m_offset(offset),
473 m_fullSize(fullSize)
474 {
475 SetDeviceOrigin(0, 0);
476 }
477
478 virtual void SetDeviceOrigin(wxCoord x, wxCoord y)
479 {
480 wxMemoryDCImpl::SetDeviceOrigin(x - m_offset.x, y - m_offset.y);
481 }
482
483 virtual void DoGetDeviceOrigin(wxCoord *x, wxCoord *y) const
484 {
485 wxMemoryDCImpl::DoGetDeviceOrigin(x, y);
486 if ( x ) *x += m_offset.x;
487 if ( x ) *y += m_offset.y;
488 }
489
490 virtual void DoGetSize(int *width, int *height) const
491 {
492 if ( width )
493 *width = m_fullSize.x;
494 if ( height )
495 *height = m_fullSize.y;
496 }
497
498 private:
499 wxPoint m_offset;
500 wxSize m_fullSize;
501 };
502
503 class PageFragmentDC : public wxDC
504 {
505 public:
506 PageFragmentDC(wxDC* printer, wxBitmap& bmp,
507 const wxPoint& offset,
508 const wxSize& fullSize)
509 : wxDC(new PageFragmentDCImpl((wxMemoryDC*)this, printer, offset, fullSize))
510 {
511 static_cast<PageFragmentDCImpl*>(m_pimpl)->DoSelect(bmp);
512 }
513 };
514
515 // estimate how big chunks we can render, given available RAM
516 long 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
567 bool 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 {
592 PageFragmentDC memoryDC(&printer, large,
593 rect.GetPosition(),
594 wxSize(m_pageWidth, m_pageHeight));
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
637 bool 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
699 bool 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
723 /****************************************************************************
724
725 FUNCTION: wxAbortProc()
726
727 PURPOSE: Processes messages for the Abort Dialog box
728
729 ****************************************************************************/
730
731 LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
732 {
733 MSG msg;
734
735 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
736 return(TRUE);
737
738 /* Process messages intended for the abort dialog box */
739
740 while (!wxPrinterBase::sm_abortIt && ::PeekMessage(&msg, 0, 0, 0, TRUE))
741 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
742 TranslateMessage(&msg);
743 DispatchMessage(&msg);
744 }
745
746 /* bAbort is TRUE (return is FALSE) if the user has aborted */
747
748 return !wxPrinterBase::sm_abortIt;
749 }
750
751 #endif
752 // wxUSE_PRINTING_ARCHITECTURE