]> git.saurik.com Git - wxWidgets.git/blob - src/msw/printwin.cpp
1. frames respect update region (Tom Surace patch)
[wxWidgets.git] / src / msw / printwin.cpp
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$
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ===========================================================================
13 // declarations
14 // ===========================================================================
15
16 // ---------------------------------------------------------------------------
17 // headers
18 // ---------------------------------------------------------------------------
19
20 #ifdef __GNUG__
21 #pragma implementation "printwin.h"
22 #endif
23
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
26
27 #ifdef __BORLANDC__
28 #pragma hdrstop
29 #endif
30
31 #include "wx/defs.h"
32
33 #ifndef WX_PRECOMP
34 #include "wx/window.h"
35 #include "wx/msw/private.h"
36 #include "wx/utils.h"
37 #include "wx/dc.h"
38 #include "wx/app.h"
39 #include "wx/msgdlg.h"
40 #include "wx/intl.h"
41 #endif
42
43 #include "wx/msw/printwin.h"
44 #include "wx/dcprint.h"
45 #include "wx/printdlg.h"
46 #include "wx/log.h"
47 #include "wx/msw/private.h"
48
49 #include <stdlib.h>
50
51 #include "wx/msw/private.h"
52
53 #include <commdlg.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 #if !USE_SHARED_LIBRARY
70 IMPLEMENT_DYNAMIC_CLASS(wxWindowsPrinter, wxPrinterBase)
71 IMPLEMENT_CLASS(wxWindowsPrintPreview, wxPrintPreviewBase)
72 #endif
73
74 // ===========================================================================
75 // implementation
76 // ===========================================================================
77
78 // ---------------------------------------------------------------------------
79 // Printer
80 // ---------------------------------------------------------------------------
81
82 wxWindowsPrinter::wxWindowsPrinter(wxPrintDialogData *data)
83 : wxPrinterBase(data)
84 {
85 m_lpAbortProc = (WXFARPROC) MakeProcInstance((FARPROC) wxAbortProc, wxGetInstance());
86 }
87
88 wxWindowsPrinter::~wxWindowsPrinter()
89 {
90 FreeProcInstance((FARPROC) m_lpAbortProc);
91 }
92
93 bool wxWindowsPrinter::Print(wxWindow *parent, wxPrintout *printout, bool prompt)
94 {
95 sm_abortIt = FALSE;
96 sm_abortWindow = NULL;
97
98 if (!printout)
99 return FALSE;
100
101 printout->SetIsPreview(FALSE);
102 printout->OnPreparePrinting();
103
104 // Get some parameters from the printout, if defined
105 int fromPage, toPage;
106 int minPage, maxPage;
107 printout->GetPageInfo(&minPage, &maxPage, &fromPage, &toPage);
108
109 if (maxPage == 0)
110 return FALSE;
111
112 m_printDialogData.SetMinPage(minPage);
113 m_printDialogData.SetMaxPage(maxPage);
114 if (fromPage != 0)
115 m_printDialogData.SetFromPage(fromPage);
116 if (toPage != 0)
117 m_printDialogData.SetToPage(toPage);
118
119 if (minPage != 0)
120 {
121 m_printDialogData.EnablePageNumbers(TRUE);
122 if (m_printDialogData.GetFromPage() < m_printDialogData.GetMinPage())
123 m_printDialogData.SetFromPage(m_printDialogData.GetMinPage());
124 else if (m_printDialogData.GetFromPage() > m_printDialogData.GetMaxPage())
125 m_printDialogData.SetFromPage(m_printDialogData.GetMaxPage());
126 if (m_printDialogData.GetToPage() > m_printDialogData.GetMaxPage())
127 m_printDialogData.SetToPage(m_printDialogData.GetMaxPage());
128 else if (m_printDialogData.GetToPage() < m_printDialogData.GetMinPage())
129 m_printDialogData.SetToPage(m_printDialogData.GetMinPage());
130 }
131 else
132 m_printDialogData.EnablePageNumbers(FALSE);
133
134 // Create a suitable device context
135 wxDC *dc = NULL;
136 if (prompt)
137 {
138 dc = PrintDialog(parent);
139 if (!dc)
140 return FALSE;
141 }
142 else
143 {
144 // dc = new wxPrinterDC("", "", "", FALSE, m_printData.GetOrientation());
145 dc = new wxPrinterDC(m_printDialogData.GetPrintData());
146 }
147
148 // May have pressed cancel.
149 if (!dc || !dc->Ok())
150 {
151 if (dc) delete dc;
152 return FALSE;
153 }
154
155 int logPPIScreenX = 0;
156 int logPPIScreenY = 0;
157 int logPPIPrinterX = 0;
158 int logPPIPrinterY = 0;
159
160 HDC hdc = ::GetDC(NULL);
161 logPPIScreenX = ::GetDeviceCaps(hdc, LOGPIXELSX);
162 logPPIScreenY = ::GetDeviceCaps(hdc, LOGPIXELSY);
163 ::ReleaseDC(NULL, hdc);
164
165 logPPIPrinterX = ::GetDeviceCaps((HDC) dc->GetHDC(), LOGPIXELSX);
166 logPPIPrinterY = ::GetDeviceCaps((HDC) dc->GetHDC(), LOGPIXELSY);
167 if (logPPIPrinterX == 0 || logPPIPrinterY == 0)
168 {
169 delete dc;
170 return FALSE;
171 }
172
173 printout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
174 printout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
175
176 // Set printout parameters
177 printout->SetDC(dc);
178
179 int w, h;
180 dc->GetSize(&w, &h);
181 printout->SetPageSizePixels((int)w, (int)h);
182
183 dc->GetSizeMM(&w, &h);
184 printout->SetPageSizeMM((int)w, (int)h);
185
186 // Create an abort window
187 wxBeginBusyCursor();
188
189 wxWindow *win = CreateAbortWindow(parent, printout);
190 wxYield();
191
192 #if defined(__BORLANDC__) || defined(__GNUWIN32__) || defined(__SALFORDC__) || !defined(__WIN32__)
193 #ifdef STRICT
194 ::SetAbortProc((HDC) dc->GetHDC(), (ABORTPROC) m_lpAbortProc);
195 #else
196 ::SetAbortProc((HDC) dc->GetHDC(), (FARPROC) m_lpAbortProc);
197 #endif
198 #else
199 ::SetAbortProc((HDC) dc->GetHDC(), (int (_stdcall *)
200 // cast it to right type only if required
201 // FIXME it's really cdecl and we're casting it to stdcall - either there is
202 // something I don't understand or it will crash at first usage
203 #ifdef STRICT
204 (HDC, int)
205 #else
206 ()
207 #endif
208 )m_lpAbortProc);
209 #endif
210
211 if (!win)
212 {
213 wxEndBusyCursor();
214 wxLogDebug(_T("Could not create an abort dialog."));
215
216 delete dc;
217 }
218 sm_abortWindow = win;
219 sm_abortWindow->Show(TRUE);
220 wxSafeYield();
221
222 printout->OnBeginPrinting();
223
224 bool keepGoing = TRUE;
225
226 int copyCount;
227 for (copyCount = 1; copyCount <= m_printDialogData.GetNoCopies(); copyCount ++)
228 {
229 if (!printout->OnBeginDocument(m_printDialogData.GetFromPage(), m_printDialogData.GetToPage()))
230 {
231 wxEndBusyCursor();
232 wxLogError(_("Could not start printing."));
233 break;
234 }
235 if (sm_abortIt)
236 break;
237
238 int pn;
239 for (pn = m_printDialogData.GetFromPage(); keepGoing && (pn <= m_printDialogData.GetToPage()) && printout->HasPage(pn);
240 pn++)
241 {
242 if (sm_abortIt)
243 {
244 keepGoing = FALSE;
245 break;
246 }
247 else
248 {
249 dc->StartPage();
250 printout->OnPrintPage(pn);
251 dc->EndPage();
252 }
253 }
254 printout->OnEndDocument();
255 }
256
257 printout->OnEndPrinting();
258
259 if (sm_abortWindow)
260 {
261 sm_abortWindow->Show(FALSE);
262 delete sm_abortWindow;
263 sm_abortWindow = NULL;
264 }
265
266 wxEndBusyCursor();
267
268 delete dc;
269
270 return TRUE;
271 }
272
273 wxDC* wxWindowsPrinter::PrintDialog(wxWindow *parent)
274 {
275 wxDC* dc = (wxDC*) NULL;
276
277 wxPrintDialog dialog(parent, & m_printDialogData);
278 int ret = dialog.ShowModal();
279
280 if (ret == wxID_OK)
281 {
282 dc = dialog.GetPrintDC();
283 m_printDialogData = dialog.GetPrintDialogData();
284 }
285
286 return dc;
287 }
288
289 bool wxWindowsPrinter::Setup(wxWindow *parent)
290 {
291 wxPrintDialog dialog(parent, & m_printDialogData);
292 dialog.GetPrintDialogData().SetSetupDialog(TRUE);
293
294 int ret = dialog.ShowModal();
295
296 if (ret == wxID_OK)
297 {
298 m_printDialogData = dialog.GetPrintDialogData();
299 }
300
301 return (ret == wxID_OK);
302 }
303
304 /*
305 * Print preview
306 */
307
308 wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
309 wxPrintout *printoutForPrinting,
310 wxPrintDialogData *data)
311 : wxPrintPreviewBase(printout, printoutForPrinting, data)
312 {
313 DetermineScaling();
314 }
315
316 wxWindowsPrintPreview::wxWindowsPrintPreview(wxPrintout *printout,
317 wxPrintout *printoutForPrinting,
318 wxPrintData *data)
319 : wxPrintPreviewBase(printout, printoutForPrinting, data)
320 {
321 DetermineScaling();
322 }
323
324 wxWindowsPrintPreview::~wxWindowsPrintPreview()
325 {
326 }
327
328 bool wxWindowsPrintPreview::Print(bool interactive)
329 {
330 if (!m_printPrintout)
331 return FALSE;
332 wxWindowsPrinter printer(&m_printDialogData);
333 return printer.Print(m_previewFrame, m_printPrintout, interactive);
334 }
335
336 void wxWindowsPrintPreview::DetermineScaling()
337 {
338 HDC dc = ::GetDC(NULL);
339 int screenWidth = ::GetDeviceCaps(dc, HORZSIZE);
340 // int screenHeight = ::GetDeviceCaps(dc, VERTSIZE);
341 int screenXRes = ::GetDeviceCaps(dc, HORZRES);
342 // int screenYRes = ::GetDeviceCaps(dc, VERTRES);
343 int logPPIScreenX = ::GetDeviceCaps(dc, LOGPIXELSX);
344 int logPPIScreenY = ::GetDeviceCaps(dc, LOGPIXELSY);
345 m_previewPrintout->SetPPIScreen(logPPIScreenX, logPPIScreenY);
346
347 ::ReleaseDC(NULL, dc);
348
349 // Get a device context for the currently selected printer
350 wxPrinterDC printerDC(m_printDialogData.GetPrintData());
351
352 int printerWidth = 150;
353 int printerHeight = 250;
354 int printerXRes = 1500;
355 int printerYRes = 2500;
356
357 if (printerDC.GetHDC())
358 {
359 printerWidth = ::GetDeviceCaps((HDC) printerDC.GetHDC(), HORZSIZE);
360 printerHeight = ::GetDeviceCaps((HDC) printerDC.GetHDC(), VERTSIZE);
361 printerXRes = ::GetDeviceCaps((HDC) printerDC.GetHDC(), HORZRES);
362 printerYRes = ::GetDeviceCaps((HDC) printerDC.GetHDC(), VERTRES);
363
364 int logPPIPrinterX = ::GetDeviceCaps((HDC) printerDC.GetHDC(), LOGPIXELSX);
365 int logPPIPrinterY = ::GetDeviceCaps((HDC) printerDC.GetHDC(), LOGPIXELSY);
366
367 m_previewPrintout->SetPPIPrinter(logPPIPrinterX, logPPIPrinterY);
368 m_previewPrintout->SetPageSizeMM(printerWidth, printerHeight);
369
370 if (logPPIPrinterX == 0 || logPPIPrinterY == 0 || printerWidth == 0 || printerHeight == 0)
371 m_isOk = FALSE;
372 }
373 else
374 m_isOk = FALSE;
375
376 m_pageWidth = printerXRes;
377 m_pageHeight = printerYRes;
378
379 // At 100%, the page should look about page-size on the screen.
380 m_previewScale = (float)((float)screenWidth/(float)printerWidth);
381 m_previewScale = m_previewScale * (float)((float)screenXRes/(float)printerYRes);
382 }
383
384 /****************************************************************************
385
386 FUNCTION: wxAbortProc()
387
388 PURPOSE: Processes messages for the Abort Dialog box
389
390 ****************************************************************************/
391
392 LONG APIENTRY _EXPORT wxAbortProc(HDC WXUNUSED(hPr), int WXUNUSED(Code))
393 {
394 MSG msg;
395
396 if (!wxPrinterBase::sm_abortWindow) /* If the abort dialog isn't up yet */
397 return(TRUE);
398
399 /* Process messages intended for the abort dialog box */
400
401 while (!wxPrinterBase::sm_abortIt && PeekMessage(&msg, 0, 0, 0, TRUE))
402 if (!IsDialogMessage((HWND) wxPrinterBase::sm_abortWindow->GetHWND(), &msg)) {
403 TranslateMessage(&msg);
404 DispatchMessage(&msg);
405 }
406
407 /* bAbort is TRUE (return is FALSE) if the user has aborted */
408
409 return (!wxPrinterBase::sm_abortIt);
410 }
411