Send all menu item actions to a dedicated target. This is to ensure
[wxWidgets.git] / src / msw / printwin.cpp
CommitLineData
2bda0e17
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: printwin.cpp
3// Purpose: wxWindowsPrinter framework
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
6c9a19aa
JS
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
103aec29
VZ
12// ===========================================================================
13// declarations
14// ===========================================================================
15
16// ---------------------------------------------------------------------------
17// headers
18// ---------------------------------------------------------------------------
19
14f355c2 20#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
103aec29 21 #pragma implementation "printwin.h"
2bda0e17
KB
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
103aec29 28 #pragma hdrstop
2bda0e17
KB
29#endif
30
31#include "wx/defs.h"
32
12bdd77c
JS
33// Don't use the Windows printer if we're in wxUniv mode and using
34// the PostScript architecture
35#if wxUSE_PRINTING_ARCHITECTURE && (!defined(__WXUNIVERSAL__) || !wxUSE_POSTSCRIPT_ARCHITECTURE_IN_MSW)
f6bcfd97 36
2bda0e17 37#ifndef WX_PRECOMP
0c589ad0
BM
38 #include "wx/window.h"
39 #include "wx/msw/private.h"
103aec29
VZ
40 #include "wx/utils.h"
41 #include "wx/dc.h"
42 #include "wx/app.h"
43 #include "wx/msgdlg.h"
0c589ad0 44 #include "wx/intl.h"
2bda0e17
KB
45#endif
46
47#include "wx/msw/printwin.h"
48#include "wx/dcprint.h"
49#include "wx/printdlg.h"
6776a0b2 50#include "wx/log.h"
2bda0e17
KB
51#include "wx/msw/private.h"
52
53#include <stdlib.h>
2bda0e17 54
42e69d6b
VZ
55#include "wx/msw/private.h"
56
57#include <commdlg.h>
2bda0e17
KB
58
59#ifndef __WIN32__
103aec29 60 #include <print.h>
2bda0e17
KB
61#endif
62
103aec29
VZ
63// ---------------------------------------------------------------------------
64// private functions
65// ---------------------------------------------------------------------------
66
2bda0e17
KB
67LONG APIENTRY _EXPORT wxAbortProc(HDC hPr, int Code);
68
103aec29
VZ
69// ---------------------------------------------------------------------------
70// wxWin macros
71// ---------------------------------------------------------------------------
72
103aec29
VZ
73 IMPLEMENT_DYNAMIC_CLASS(wxWindowsPrinter, wxPrinterBase)
74 IMPLEMENT_CLASS(wxWindowsPrintPreview, wxPrintPreviewBase)
2bda0e17 75
103aec29
VZ
76// ===========================================================================
77// implementation
78// ===========================================================================
79
80// ---------------------------------------------------------------------------
81// Printer
82// ---------------------------------------------------------------------------
7bcb11d3 83
103aec29
VZ
84wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData *data)
85 : wxPrinterBase(data)
2bda0e17 86{
34da0970 87 m_lpAbortProc = (WXFARPROC) MakeProcInstance((FARPROC) wxAbortProc, wxGetInstance());
2bda0e17
KB
88}
89
103aec29 90wxWindowsPrinter::~wxWindowsPrinter()
2bda0e17 91{
33ac7e6f
KB
92 // avoids warning about statement with no effect (FreeProcInstance
93 // doesn't do anything under Win32)
d66dcb60 94#if !defined(__WIN32__) && !defined(__NT__)
34da0970 95 FreeProcInstance((FARPROC) m_lpAbortProc);
a17e237f 96#endif
2bda0e17
KB
97}
98
99bool wxWindowsPrinter::Print(wxWindow *parent, wxPrintout *printout, bool prompt)
100{
7bcb11d3
JS
101 sm_abortIt = FALSE;
102 sm_abortWindow = NULL;
103aec29 103
7bcb11d3 104 if (!printout)
f6bcfd97
BP
105 {
106 sm_lastError = wxPRINTER_ERROR;
7bcb11d3 107 return FALSE;
f6bcfd97 108 }
103aec29 109
7bcb11d3 110 printout->SetIsPreview(FALSE);
1044a386 111
d2b354f9
JS
112 if (m_printDialogData.GetMinPage() < 1)
113 m_printDialogData.SetMinPage(1);
114 if (m_printDialogData.GetMaxPage() < 1)
115 m_printDialogData.SetMaxPage(9999);
2bda0e17 116
103aec29 117 // Create a suitable device context
7bcb11d3
JS
118 wxDC *dc = NULL;
119 if (prompt)
120 {
121 dc = PrintDialog(parent);
122 if (!dc)
123 return FALSE;
124 }
125 else
126 {
7bcb11d3
JS
127 dc = new wxPrinterDC(m_printDialogData.GetPrintData());
128 }
103aec29 129
7bcb11d3
JS
130 // May have pressed cancel.
131 if (!dc || !dc->Ok())
132 {
133 if (dc) delete dc;
134 return FALSE;
135 }
103aec29 136
7bcb11d3
JS
137 int logPPIScreenX = 0;
138 int logPPIScreenY = 0;
139 int logPPIPrinterX = 0;
140 int logPPIPrinterY = 0;
103aec29 141
7bcb11d3
JS
142 HDC hdc = ::GetDC(NULL);
143 logPPIScreenX = ::GetDeviceCaps(hdc, LOGPIXELSX);
144 logPPIScreenY = ::GetDeviceCaps(hdc, LOGPIXELSY);
145 ::ReleaseDC(NULL, hdc);
103aec29 146
7bcb11d3
JS
147 logPPIPrinterX = ::GetDeviceCaps((HDC) dc->GetHDC(), LOGPIXELSX);
148 logPPIPrinterY = ::GetDeviceCaps((HDC) dc->GetHDC(), LOGPIXELSY);
149 if (logPPIPrinterX == 0 || logPPIPrinterY == 0)
150 {
151 delete dc;
f6bcfd97 152 sm_lastError = wxPRINTER_ERROR;
7bcb11d3
JS
153 return FALSE;
154 }
103aec29 155
7bcb11d3
JS
156 printout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
157 printout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
103aec29
VZ
158
159 // Set printout parameters
7bcb11d3 160 printout->SetDC(dc);
103aec29 161
7bcb11d3
JS
162 int w, h;
163 dc->GetSize(&w, &h);
164 printout->SetPageSizePixels((int)w, (int)h);
d6a1743b 165
7bcb11d3
JS
166 dc->GetSizeMM(&w, &h);
167 printout->SetPageSizeMM((int)w, (int)h);
103aec29 168
7bcb11d3
JS
169 // Create an abort window
170 wxBeginBusyCursor();
103aec29 171
1044a386
JS
172 printout->OnPreparePrinting();
173
d2b354f9
JS
174 // Get some parameters from the printout, if defined
175 int fromPage, toPage;
176 int minPage, maxPage;
177 printout->GetPageInfo(&minPage, &maxPage, &fromPage, &toPage);
178
179 if (maxPage == 0)
180 {
181 sm_lastError = wxPRINTER_ERROR;
182 wxEndBusyCursor();
183 return FALSE;
184 }
185
186 // Only set min and max, because from and to have been
187 // set by the user
188 m_printDialogData.SetMinPage(minPage);
189 m_printDialogData.SetMaxPage(maxPage);
190
7bcb11d3
JS
191 wxWindow *win = CreateAbortWindow(parent, printout);
192 wxYield();
103aec29 193
ce3ed50d 194#if defined(__BORLANDC__) || defined(__GNUWIN32__) || defined(__SALFORDC__) || !defined(__WIN32__)
27a9bd48
PA
195#ifdef STRICT
196 ::SetAbortProc((HDC) dc->GetHDC(), (ABORTPROC) m_lpAbortProc);
197#else
7bcb11d3 198 ::SetAbortProc((HDC) dc->GetHDC(), (FARPROC) m_lpAbortProc);
27a9bd48 199#endif
7bcb11d3
JS
200#else
201 ::SetAbortProc((HDC) dc->GetHDC(), (int (_stdcall *)
202 // cast it to right type only if required
103aec29
VZ
203 // FIXME it's really cdecl and we're casting it to stdcall - either there is
204 // something I don't understand or it will crash at first usage
7bcb11d3
JS
205#ifdef STRICT
206 (HDC, int)
d6a1743b 207#else
7bcb11d3 208 ()
d6a1743b 209#endif
7bcb11d3
JS
210 )m_lpAbortProc);
211#endif
103aec29 212
7bcb11d3 213 if (!win)
2bda0e17 214 {
7bcb11d3 215 wxEndBusyCursor();
223d09f6 216 wxLogDebug(wxT("Could not create an abort dialog."));
f6bcfd97 217 sm_lastError = wxPRINTER_ERROR;
103aec29 218
7bcb11d3 219 delete dc;
2bda0e17 220 }
7bcb11d3
JS
221 sm_abortWindow = win;
222 sm_abortWindow->Show(TRUE);
103aec29
VZ
223 wxSafeYield();
224
7bcb11d3 225 printout->OnBeginPrinting();
103aec29 226
f6bcfd97
BP
227 sm_lastError = wxPRINTER_NO_ERROR;
228
7bcb11d3 229 int copyCount;
70846f0a
VZ
230 for ( copyCount = 1;
231 copyCount <= m_printDialogData.GetNoCopies();
232 copyCount++ )
7bcb11d3
JS
233 {
234 if (!printout->OnBeginDocument(m_printDialogData.GetFromPage(), m_printDialogData.GetToPage()))
235 {
236 wxEndBusyCursor();
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;
70846f0a
VZ
248 for ( pn = m_printDialogData.GetFromPage();
249 pn <= m_printDialogData.GetToPage() && printout->HasPage(pn);
250 pn++ )
7bcb11d3 251 {
70846f0a 252 if ( sm_abortIt )
7bcb11d3 253 {
f6bcfd97 254 sm_lastError = wxPRINTER_CANCELLED;
7bcb11d3
JS
255 break;
256 }
70846f0a
VZ
257
258 dc->StartPage();
259 bool cont = printout->OnPrintPage(pn);
260 dc->EndPage();
261
262 if ( !cont )
f6bcfd97
BP
263 {
264 sm_lastError = wxPRINTER_CANCELLED;
70846f0a 265 break;
f6bcfd97 266 }
7bcb11d3 267 }
70846f0a 268
7bcb11d3
JS
269 printout->OnEndDocument();
270 }
103aec29 271
7bcb11d3 272 printout->OnEndPrinting();
103aec29 273
7bcb11d3 274 if (sm_abortWindow)
2bda0e17 275 {
7bcb11d3
JS
276 sm_abortWindow->Show(FALSE);
277 delete sm_abortWindow;
278 sm_abortWindow = NULL;
2bda0e17 279 }
103aec29 280
7bcb11d3 281 wxEndBusyCursor();
103aec29 282
7bcb11d3 283 delete dc;
103aec29 284
f6bcfd97 285 return (sm_lastError == wxPRINTER_NO_ERROR);
7bcb11d3 286}
2bda0e17 287
7bcb11d3
JS
288wxDC* wxWindowsPrinter::PrintDialog(wxWindow *parent)
289{
290 wxDC* dc = (wxDC*) NULL;
2bda0e17 291
7bcb11d3
JS
292 wxPrintDialog dialog(parent, & m_printDialogData);
293 int ret = dialog.ShowModal();
2bda0e17 294
7bcb11d3
JS
295 if (ret == wxID_OK)
296 {
297 dc = dialog.GetPrintDC();
298 m_printDialogData = dialog.GetPrintDialogData();
33ac7e6f 299 if (dc == NULL)
f6bcfd97
BP
300 sm_lastError = wxPRINTER_ERROR;
301 else
302 sm_lastError = wxPRINTER_NO_ERROR;
7bcb11d3 303 }
f6bcfd97
BP
304 else
305 sm_lastError = wxPRINTER_CANCELLED;
2bda0e17 306
7bcb11d3 307 return dc;
2bda0e17
KB
308}
309
310bool wxWindowsPrinter::Setup(wxWindow *parent)
311{
7bcb11d3
JS
312 wxPrintDialog dialog(parent, & m_printDialogData);
313 dialog.GetPrintDialogData().SetSetupDialog(TRUE);
314
315 int ret = dialog.ShowModal();
316
317 if (ret == wxID_OK)
318 {
319 m_printDialogData = dialog.GetPrintDialogData();
320 }
321
322 return (ret == wxID_OK);
2bda0e17
KB
323}
324
325/*
7bcb11d3
JS
326* Print preview
327*/
2bda0e17 328
103aec29
VZ
329wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
330 wxPrintout *printoutForPrinting,
331 wxPrintDialogData *data)
332 : wxPrintPreviewBase(printout, printoutForPrinting, data)
2bda0e17 333{
7bcb11d3 334 DetermineScaling();
2bda0e17
KB
335}
336
103aec29
VZ
337wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
338 wxPrintout *printoutForPrinting,
339 wxPrintData *data)
340 : wxPrintPreviewBase(printout, printoutForPrinting, data)
341{
342 DetermineScaling();
343}
344
345wxWindowsPrintPreview::~wxWindowsPrintPreview()
2bda0e17
KB
346{
347}
348
349bool wxWindowsPrintPreview::Print(bool interactive)
350{
7bcb11d3
JS
351 if (!m_printPrintout)
352 return FALSE;
353 wxWindowsPrinter printer(&m_printDialogData);
354 return printer.Print(m_previewFrame, m_printPrintout, interactive);
2bda0e17
KB
355}
356
103aec29 357void wxWindowsPrintPreview::DetermineScaling()
2bda0e17
KB
358{
359 HDC dc = ::GetDC(NULL);
360 int screenWidth = ::GetDeviceCaps(dc, HORZSIZE);
f4fefc23 361 int screenYRes = ::GetDeviceCaps(dc, VERTRES);
2bda0e17
KB
362 int logPPIScreenX = ::GetDeviceCaps(dc, LOGPIXELSX);
363 int logPPIScreenY = ::GetDeviceCaps(dc, LOGPIXELSY);
34da0970 364 m_previewPrintout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
103aec29 365
2bda0e17 366 ::ReleaseDC(NULL, dc);
103aec29 367
2bda0e17 368 // Get a device context for the currently selected printer
7bcb11d3 369 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
103aec29 370
2bda0e17
KB
371 int printerWidth = 150;
372 int printerHeight = 250;
373 int printerXRes = 1500;
374 int printerYRes = 2500;
103aec29 375
f4fefc23
VZ
376 dc = GetHdcOf(printerDC);
377 if ( dc )
2bda0e17 378 {
f4fefc23
VZ
379 printerWidth = ::GetDeviceCaps(dc, HORZSIZE);
380 printerHeight = ::GetDeviceCaps(dc, VERTSIZE);
381 printerXRes = ::GetDeviceCaps(dc, HORZRES);
382 printerYRes = ::GetDeviceCaps(dc, VERTRES);
103aec29 383
f4fefc23
VZ
384 int logPPIPrinterX = ::GetDeviceCaps(dc, LOGPIXELSX);
385 int logPPIPrinterY = ::GetDeviceCaps(dc, LOGPIXELSY);
103aec29 386
7bcb11d3
JS
387 m_previewPrintout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
388 m_previewPrintout->SetPageSizeMM(printerWidth, printerHeight);
103aec29 389
7bcb11d3
JS
390 if (logPPIPrinterX == 0 || logPPIPrinterY == 0 || printerWidth == 0 || printerHeight == 0)
391 m_isOk = FALSE;
2bda0e17
KB
392 }
393 else
7bcb11d3 394 m_isOk = FALSE;
103aec29 395
34da0970
JS
396 m_pageWidth = printerXRes;
397 m_pageHeight = printerYRes;
103aec29 398
2bda0e17 399 // At 100%, the page should look about page-size on the screen.
34da0970 400 m_previewScale = (float)((float)screenWidth/(float)printerWidth);
f4fefc23 401 m_previewScale = m_previewScale * (float)((float)screenYRes/(float)printerYRes);
2bda0e17
KB
402}
403
404/****************************************************************************
405
7bcb11d3 406 FUNCTION: wxAbortProc()
103aec29 407
2bda0e17 408 PURPOSE: Processes messages for the Abort Dialog box
103aec29 409
2bda0e17
KB
410****************************************************************************/
411
412LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
413{
414 MSG msg;
103aec29 415
34da0970 416 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
2bda0e17 417 return(TRUE);
103aec29 418
2bda0e17 419 /* Process messages intended for the abort dialog box */
103aec29 420
34da0970
JS
421 while (!wxPrinterBase::sm_abortIt && PeekMessage(&msg, 0, 0, 0, TRUE))
422 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
2bda0e17
KB
423 TranslateMessage(&msg);
424 DispatchMessage(&msg);
425 }
103aec29 426
7bcb11d3 427 /* bAbort is TRUE (return is FALSE) if the user has aborted */
103aec29 428
7bcb11d3 429 return (!wxPrinterBase::sm_abortIt);
2bda0e17
KB
430}
431
f6bcfd97
BP
432#endif
433 // wxUSE_PRINTING_ARCHITECTURE