]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: samples/printing.cpp | |
3 | // Purpose: Printing demo for wxWidgets | |
4 | // Author: Julian Smart | |
5 | // Modified by: | |
6 | // Created: 1995 | |
7 | // RCS-ID: $Id$ | |
8 | // Copyright: (c) Julian Smart | |
9 | // Licence: wxWindows licence | |
10 | ///////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | // For compilers that support precompilation, includes "wx/wx.h". | |
13 | #include "wx/wxprec.h" | |
14 | ||
15 | #ifdef __BORLANDC__ | |
16 | #pragma hdrstop | |
17 | #endif | |
18 | ||
19 | #ifndef WX_PRECOMP | |
20 | #include "wx/wx.h" | |
21 | #endif | |
22 | ||
23 | #if !wxUSE_PRINTING_ARCHITECTURE | |
24 | #error "You must set wxUSE_PRINTING_ARCHITECTURE to 1 in setup.h, and recompile the library." | |
25 | #endif | |
26 | ||
27 | // Set this to 1 if you want to test PostScript printing under MSW. | |
28 | // However, you'll also need to edit src/msw/makefile.nt. | |
29 | #define wxTEST_POSTSCRIPT_IN_MSW 0 | |
30 | ||
31 | #include <ctype.h> | |
32 | #include "wx/metafile.h" | |
33 | #include "wx/print.h" | |
34 | #include "wx/printdlg.h" | |
35 | #include "wx/image.h" | |
36 | #include "wx/accel.h" | |
37 | ||
38 | #if wxTEST_POSTSCRIPT_IN_MSW | |
39 | #include "wx/generic/printps.h" | |
40 | #include "wx/generic/prntdlgg.h" | |
41 | #endif | |
42 | ||
43 | #ifdef __WXMAC__ | |
44 | #include "wx/mac/printdlg.h" | |
45 | #endif | |
46 | ||
47 | #include "printing.h" | |
48 | ||
49 | #ifndef __WXMSW__ | |
50 | #include "mondrian.xpm" | |
51 | #endif | |
52 | ||
53 | #if wxUSE_LIBGNOMEPRINT | |
54 | #include "wx/html/forcelnk.h" | |
55 | FORCE_LINK(gnome_print) | |
56 | #endif | |
57 | ||
58 | ||
59 | // Declare a frame | |
60 | MyFrame *frame = (MyFrame *) NULL; | |
61 | // int orientation = wxPORTRAIT; | |
62 | ||
63 | // Global print data, to remember settings during the session | |
64 | wxPrintData *g_printData = (wxPrintData*) NULL ; | |
65 | ||
66 | // Global page setup data | |
67 | wxPageSetupDialogData* g_pageSetupData = (wxPageSetupDialogData*) NULL; | |
68 | ||
69 | // Main proc | |
70 | IMPLEMENT_APP(MyApp) | |
71 | ||
72 | // Writes a header on a page. Margin units are in millimetres. | |
73 | bool WritePageHeader(wxPrintout *printout, wxDC *dc, const wxChar *text, float mmToLogical); | |
74 | ||
75 | // The `main program' equivalent, creating the windows and returning the | |
76 | // main frame | |
77 | ||
78 | bool MyApp::OnInit(void) | |
79 | { | |
80 | if ( !wxApp::OnInit() ) | |
81 | return false; | |
82 | ||
83 | wxInitAllImageHandlers(); | |
84 | ||
85 | m_testFont.Create(10, wxSWISS, wxNORMAL, wxNORMAL); | |
86 | ||
87 | g_printData = new wxPrintData; | |
88 | // You could set an initial paper size here | |
89 | // g_printData->SetPaperId(wxPAPER_LETTER); // for Americans | |
90 | // g_printData->SetPaperId(wxPAPER_A4); // for everyone else | |
91 | ||
92 | g_pageSetupData = new wxPageSetupDialogData; | |
93 | // copy over initial paper size from print record | |
94 | (*g_pageSetupData) = *g_printData; | |
95 | // Set some initial page margins in mm. | |
96 | g_pageSetupData->SetMarginTopLeft(wxPoint(15, 15)); | |
97 | g_pageSetupData->SetMarginBottomRight(wxPoint(15, 15)); | |
98 | ||
99 | // Create the main frame window | |
100 | frame = new MyFrame((wxFrame *) NULL, _T("wxWidgets Printing Demo"), | |
101 | wxPoint(0, 0), wxSize(400, 400)); | |
102 | ||
103 | #if wxUSE_STATUSBAR | |
104 | // Give it a status line | |
105 | frame->CreateStatusBar(2); | |
106 | #endif // wxUSE_STATUSBAR | |
107 | ||
108 | // Load icon and bitmap | |
109 | frame->SetIcon( wxICON( mondrian) ); | |
110 | ||
111 | // Make a menubar | |
112 | wxMenu *file_menu = new wxMenu; | |
113 | ||
114 | file_menu->Append(WXPRINT_PRINT, _T("&Print..."), _T("Print")); | |
115 | file_menu->Append(WXPRINT_PAGE_SETUP, _T("Page Set&up..."), _T("Page setup")); | |
116 | #ifdef __WXMAC__ | |
117 | file_menu->Append(WXPRINT_PAGE_MARGINS, _T("Page Margins..."), _T("Page margins")); | |
118 | #endif | |
119 | file_menu->Append(WXPRINT_PREVIEW, _T("Print Pre&view"), _T("Preview")); | |
120 | ||
121 | #if wxUSE_ACCEL | |
122 | // Accelerators | |
123 | wxAcceleratorEntry entries[1]; | |
124 | entries[0].Set(wxACCEL_CTRL, (int) 'V', WXPRINT_PREVIEW); | |
125 | wxAcceleratorTable accel(1, entries); | |
126 | frame->SetAcceleratorTable(accel); | |
127 | #endif | |
128 | ||
129 | #if defined(__WXMSW__) && wxTEST_POSTSCRIPT_IN_MSW | |
130 | file_menu->AppendSeparator(); | |
131 | file_menu->Append(WXPRINT_PRINT_PS, _T("Print PostScript..."), _T("Print (PostScript)")); | |
132 | file_menu->Append(WXPRINT_PAGE_SETUP_PS, _T("Page Setup PostScript..."), _T("Page setup (PostScript)")); | |
133 | file_menu->Append(WXPRINT_PREVIEW_PS, _T("Print Preview PostScript"), _T("Preview (PostScript)")); | |
134 | #endif | |
135 | ||
136 | file_menu->AppendSeparator(); | |
137 | file_menu->Append(WXPRINT_ANGLEUP, _T("Angle up\tAlt-U"), _T("Raise rotated text angle")); | |
138 | file_menu->Append(WXPRINT_ANGLEDOWN, _T("Angle down\tAlt-D"), _T("Lower rotated text angle")); | |
139 | file_menu->AppendSeparator(); | |
140 | file_menu->Append(WXPRINT_QUIT, _T("E&xit"), _T("Exit program")); | |
141 | ||
142 | wxMenu *help_menu = new wxMenu; | |
143 | help_menu->Append(WXPRINT_ABOUT, _T("&About"), _T("About this demo")); | |
144 | ||
145 | wxMenuBar *menu_bar = new wxMenuBar; | |
146 | ||
147 | menu_bar->Append(file_menu, _T("&File")); | |
148 | menu_bar->Append(help_menu, _T("&Help")); | |
149 | ||
150 | // Associate the menu bar with the frame | |
151 | frame->SetMenuBar(menu_bar); | |
152 | ||
153 | MyCanvas *canvas = new MyCanvas(frame, wxPoint(0, 0), wxSize(100, 100), wxRETAINED|wxHSCROLL|wxVSCROLL); | |
154 | ||
155 | // Give it scrollbars: the virtual canvas is 20 * 50 = 1000 pixels in each direction | |
156 | canvas->SetScrollbars(20, 20, 50, 50); | |
157 | ||
158 | frame->canvas = canvas; | |
159 | ||
160 | frame->Centre(wxBOTH); | |
161 | frame->Show(); | |
162 | ||
163 | #if wxUSE_STATUSBAR | |
164 | frame->SetStatusText(_T("Printing demo")); | |
165 | #endif // wxUSE_STATUSBAR | |
166 | ||
167 | SetTopWindow(frame); | |
168 | ||
169 | return true; | |
170 | } | |
171 | ||
172 | int MyApp::OnExit() | |
173 | { | |
174 | delete g_printData; | |
175 | delete g_pageSetupData; | |
176 | return 1; | |
177 | } | |
178 | ||
179 | BEGIN_EVENT_TABLE(MyFrame, wxFrame) | |
180 | EVT_MENU(WXPRINT_QUIT, MyFrame::OnExit) | |
181 | EVT_MENU(WXPRINT_PRINT, MyFrame::OnPrint) | |
182 | EVT_MENU(WXPRINT_PREVIEW, MyFrame::OnPrintPreview) | |
183 | EVT_MENU(WXPRINT_PAGE_SETUP, MyFrame::OnPageSetup) | |
184 | EVT_MENU(WXPRINT_ABOUT, MyFrame::OnPrintAbout) | |
185 | #if defined(__WXMSW__) && wxTEST_POSTSCRIPT_IN_MSW | |
186 | EVT_MENU(WXPRINT_PRINT_PS, MyFrame::OnPrintPS) | |
187 | EVT_MENU(WXPRINT_PREVIEW_PS, MyFrame::OnPrintPreviewPS) | |
188 | EVT_MENU(WXPRINT_PAGE_SETUP_PS, MyFrame::OnPageSetupPS) | |
189 | #endif | |
190 | #ifdef __WXMAC__ | |
191 | EVT_MENU(WXPRINT_PAGE_MARGINS, MyFrame::OnPageMargins) | |
192 | #endif | |
193 | EVT_MENU(WXPRINT_ANGLEUP, MyFrame::OnAngleUp) | |
194 | EVT_MENU(WXPRINT_ANGLEDOWN, MyFrame::OnAngleDown) | |
195 | END_EVENT_TABLE() | |
196 | ||
197 | // Define my frame constructor | |
198 | MyFrame::MyFrame(wxFrame *frame, const wxString& title, const wxPoint& pos, const wxSize& size): | |
199 | wxFrame(frame, wxID_ANY, title, pos, size) | |
200 | { | |
201 | canvas = NULL; | |
202 | m_angle = 30; | |
203 | #if 0 | |
204 | wxImage image( wxT("test.jpg") ); | |
205 | image.SetAlpha(); | |
206 | int i,j; | |
207 | for (i = 0; i < image.GetWidth(); i++) | |
208 | for (j = 0; j < image.GetHeight(); j++) | |
209 | image.SetAlpha( i, j, 50 ); | |
210 | m_bitmap = image; | |
211 | #endif | |
212 | } | |
213 | ||
214 | void MyFrame::OnExit(wxCommandEvent& WXUNUSED(event)) | |
215 | { | |
216 | Close(true /*force closing*/); | |
217 | } | |
218 | ||
219 | void MyFrame::OnPrint(wxCommandEvent& WXUNUSED(event)) | |
220 | { | |
221 | wxPrintDialogData printDialogData(* g_printData); | |
222 | ||
223 | wxPrinter printer(& printDialogData); | |
224 | MyPrintout printout(_T("My printout")); | |
225 | if (!printer.Print(this, &printout, true /*prompt*/)) | |
226 | { | |
227 | if (wxPrinter::GetLastError() == wxPRINTER_ERROR) | |
228 | wxMessageBox(_T("There was a problem printing.\nPerhaps your current printer is not set correctly?"), _T("Printing"), wxOK); | |
229 | else | |
230 | wxMessageBox(_T("You canceled printing"), _T("Printing"), wxOK); | |
231 | } | |
232 | else | |
233 | { | |
234 | (*g_printData) = printer.GetPrintDialogData().GetPrintData(); | |
235 | } | |
236 | } | |
237 | ||
238 | void MyFrame::OnPrintPreview(wxCommandEvent& WXUNUSED(event)) | |
239 | { | |
240 | // Pass two printout objects: for preview, and possible printing. | |
241 | wxPrintDialogData printDialogData(* g_printData); | |
242 | wxPrintPreview *preview = new wxPrintPreview(new MyPrintout, new MyPrintout, & printDialogData); | |
243 | if (!preview->Ok()) | |
244 | { | |
245 | delete preview; | |
246 | wxMessageBox(_T("There was a problem previewing.\nPerhaps your current printer is not set correctly?"), _T("Previewing"), wxOK); | |
247 | return; | |
248 | } | |
249 | ||
250 | wxPreviewFrame *frame = new wxPreviewFrame(preview, this, _T("Demo Print Preview"), wxPoint(100, 100), wxSize(600, 650)); | |
251 | frame->Centre(wxBOTH); | |
252 | frame->Initialize(); | |
253 | frame->Show(); | |
254 | } | |
255 | ||
256 | void MyFrame::OnPageSetup(wxCommandEvent& WXUNUSED(event)) | |
257 | { | |
258 | (*g_pageSetupData) = *g_printData; | |
259 | ||
260 | wxPageSetupDialog pageSetupDialog(this, g_pageSetupData); | |
261 | pageSetupDialog.ShowModal(); | |
262 | ||
263 | (*g_printData) = pageSetupDialog.GetPageSetupDialogData().GetPrintData(); | |
264 | (*g_pageSetupData) = pageSetupDialog.GetPageSetupDialogData(); | |
265 | } | |
266 | ||
267 | #if defined(__WXMSW__) && wxTEST_POSTSCRIPT_IN_MSW | |
268 | void MyFrame::OnPrintPS(wxCommandEvent& WXUNUSED(event)) | |
269 | { | |
270 | wxPostScriptPrinter printer(g_printData); | |
271 | MyPrintout printout(_T("My printout")); | |
272 | printer.Print(this, &printout, true/*prompt*/); | |
273 | ||
274 | (*g_printData) = printer.GetPrintData(); | |
275 | } | |
276 | ||
277 | void MyFrame::OnPrintPreviewPS(wxCommandEvent& WXUNUSED(event)) | |
278 | { | |
279 | // Pass two printout objects: for preview, and possible printing. | |
280 | wxPrintDialogData printDialogData(* g_printData); | |
281 | wxPrintPreview *preview = new wxPrintPreview(new MyPrintout, new MyPrintout, & printDialogData); | |
282 | wxPreviewFrame *frame = new wxPreviewFrame(preview, this, _T("Demo Print Preview"), wxPoint(100, 100), wxSize(600, 650)); | |
283 | frame->Centre(wxBOTH); | |
284 | frame->Initialize(); | |
285 | frame->Show(); | |
286 | } | |
287 | ||
288 | void MyFrame::OnPageSetupPS(wxCommandEvent& WXUNUSED(event)) | |
289 | { | |
290 | (*g_pageSetupData) = * g_printData; | |
291 | ||
292 | wxGenericPageSetupDialog pageSetupDialog(this, g_pageSetupData); | |
293 | pageSetupDialog.ShowModal(); | |
294 | ||
295 | (*g_printData) = pageSetupDialog.GetPageSetupDialogData().GetPrintData(); | |
296 | (*g_pageSetupData) = pageSetupDialog.GetPageSetupDialogData(); | |
297 | } | |
298 | #endif | |
299 | ||
300 | ||
301 | #ifdef __WXMAC__ | |
302 | void MyFrame::OnPageMargins(wxCommandEvent& WXUNUSED(event)) | |
303 | { | |
304 | (*g_pageSetupData) = *g_printData; | |
305 | ||
306 | wxMacPageMarginsDialog pageMarginsDialog(this, g_pageSetupData); | |
307 | pageMarginsDialog.ShowModal(); | |
308 | ||
309 | (*g_printData) = pageMarginsDialog.GetPageSetupDialogData().GetPrintData(); | |
310 | (*g_pageSetupData) = pageMarginsDialog.GetPageSetupDialogData(); | |
311 | } | |
312 | #endif | |
313 | ||
314 | ||
315 | void MyFrame::OnPrintAbout(wxCommandEvent& WXUNUSED(event)) | |
316 | { | |
317 | (void)wxMessageBox(_T("wxWidgets printing demo\nAuthor: Julian Smart"), | |
318 | _T("About wxWidgets printing demo"), wxOK|wxCENTRE); | |
319 | } | |
320 | ||
321 | void MyFrame::OnAngleUp(wxCommandEvent& WXUNUSED(event)) | |
322 | { | |
323 | m_angle += 5; | |
324 | canvas->Refresh(); | |
325 | } | |
326 | ||
327 | void MyFrame::OnAngleDown(wxCommandEvent& WXUNUSED(event)) | |
328 | { | |
329 | m_angle -= 5; | |
330 | canvas->Refresh(); | |
331 | } | |
332 | ||
333 | void MyFrame::Draw(wxDC& dc) | |
334 | { | |
335 | // This routine just draws a bunch of random stuff on the screen so that we | |
336 | // can check that different types of object are being drawn consistently | |
337 | // between the screen image, the print preview image (at various zoom | |
338 | // levels), and the printed page. | |
339 | dc.SetBackground(*wxWHITE_BRUSH); | |
340 | dc.Clear(); | |
341 | dc.SetFont(wxGetApp().m_testFont); | |
342 | ||
343 | dc.SetBackgroundMode(wxTRANSPARENT); | |
344 | ||
345 | dc.SetPen(*wxBLACK_PEN); | |
346 | dc.SetBrush(*wxLIGHT_GREY_BRUSH); | |
347 | dc.DrawRectangle(0, 0, 230, 350); | |
348 | dc.DrawLine(0, 0, 229, 349); | |
349 | dc.DrawLine(229, 0, 0, 349); | |
350 | dc.SetBrush(*wxTRANSPARENT_BRUSH); | |
351 | ||
352 | dc.SetBrush(*wxCYAN_BRUSH); | |
353 | dc.SetPen(*wxRED_PEN); | |
354 | ||
355 | dc.DrawRoundedRectangle(0, 20, 200, 80, 20); | |
356 | ||
357 | dc.DrawText( wxT("Rectangle 200 by 80"), 40, 40); | |
358 | ||
359 | dc.SetPen( wxPen(*wxBLACK,0,wxDOT_DASH) ); | |
360 | dc.DrawEllipse(50, 140, 100, 50); | |
361 | dc.SetPen(*wxRED_PEN); | |
362 | ||
363 | dc.DrawText( wxT("Test message: this is in 10 point text"), 10, 180); | |
364 | ||
365 | #if wxUSE_UNICODE | |
366 | char *test = "Hebrew שלום -- Japanese (日本語)"; | |
367 | wxString tmp = wxConvUTF8.cMB2WC( test ); | |
368 | dc.DrawText( tmp, 10, 200 ); | |
369 | #endif | |
370 | ||
371 | wxPoint points[5]; | |
372 | points[0].x = 0; | |
373 | points[0].y = 0; | |
374 | points[1].x = 20; | |
375 | points[1].y = 0; | |
376 | points[2].x = 20; | |
377 | points[2].y = 20; | |
378 | points[3].x = 10; | |
379 | points[3].y = 20; | |
380 | points[4].x = 10; | |
381 | points[4].y = -20; | |
382 | dc.DrawPolygon( 5, points, 20, 250, wxODDEVEN_RULE ); | |
383 | dc.DrawPolygon( 5, points, 50, 250, wxWINDING_RULE ); | |
384 | ||
385 | dc.DrawEllipticArc( 80, 250, 60, 30, 0.0, 270.0 ); | |
386 | ||
387 | points[0].x = 150; | |
388 | points[0].y = 250; | |
389 | points[1].x = 180; | |
390 | points[1].y = 250; | |
391 | points[2].x = 180; | |
392 | points[2].y = 220; | |
393 | points[3].x = 200; | |
394 | points[3].y = 220; | |
395 | dc.DrawSpline( 4, points ); | |
396 | ||
397 | dc.DrawArc( 20,10, 10,10, 25,40 ); | |
398 | ||
399 | wxString str; | |
400 | int i = 0; | |
401 | str.Printf( wxT("---- Text at angle %d ----"), i ); | |
402 | dc.DrawRotatedText( str, 100, 300, i ); | |
403 | ||
404 | i = m_angle; | |
405 | str.Printf( wxT("---- Text at angle %d ----"), i ); | |
406 | dc.DrawRotatedText( str, 100, 300, i ); | |
407 | ||
408 | wxIcon my_icon = wxICON(mondrian) ; | |
409 | ||
410 | dc.DrawIcon( my_icon, 100, 100); | |
411 | ||
412 | if (m_bitmap.Ok()) | |
413 | dc.DrawBitmap( m_bitmap, 10, 10 ); | |
414 | } | |
415 | ||
416 | void MyFrame::OnSize(wxSizeEvent& event ) | |
417 | { | |
418 | wxFrame::OnSize(event); | |
419 | } | |
420 | ||
421 | BEGIN_EVENT_TABLE(MyCanvas, wxScrolledWindow) | |
422 | EVT_MOUSE_EVENTS(MyCanvas::OnEvent) | |
423 | END_EVENT_TABLE() | |
424 | ||
425 | MyCanvas::MyCanvas(wxFrame *frame, const wxPoint& pos, const wxSize& size, long style): | |
426 | wxScrolledWindow(frame, wxID_ANY, pos, size, style) | |
427 | { | |
428 | SetBackgroundColour(* wxWHITE); | |
429 | } | |
430 | ||
431 | void MyCanvas::OnDraw(wxDC& dc) | |
432 | { | |
433 | frame->Draw(dc); | |
434 | } | |
435 | ||
436 | void MyCanvas::OnEvent(wxMouseEvent& WXUNUSED(event)) | |
437 | { | |
438 | } | |
439 | ||
440 | bool MyPrintout::OnPrintPage(int page) | |
441 | { | |
442 | wxDC *dc = GetDC(); | |
443 | if (dc) | |
444 | { | |
445 | if (page == 1) | |
446 | DrawPageOne(); | |
447 | else if (page == 2) | |
448 | DrawPageTwo(); | |
449 | ||
450 | // Draw page numbers at top left corner of printable area, sized so that | |
451 | // screen size of text matches paper size. | |
452 | MapScreenSizeToPage(); | |
453 | wxChar buf[200]; | |
454 | wxSprintf(buf, wxT("PAGE %d"), page); | |
455 | dc->DrawText(buf, 0, 0); | |
456 | ||
457 | return true; | |
458 | } | |
459 | else | |
460 | return false; | |
461 | } | |
462 | ||
463 | bool MyPrintout::OnBeginDocument(int startPage, int endPage) | |
464 | { | |
465 | if (!wxPrintout::OnBeginDocument(startPage, endPage)) | |
466 | return false; | |
467 | ||
468 | return true; | |
469 | } | |
470 | ||
471 | void MyPrintout::GetPageInfo(int *minPage, int *maxPage, int *selPageFrom, int *selPageTo) | |
472 | { | |
473 | *minPage = 1; | |
474 | *maxPage = 2; | |
475 | *selPageFrom = 1; | |
476 | *selPageTo = 2; | |
477 | } | |
478 | ||
479 | bool MyPrintout::HasPage(int pageNum) | |
480 | { | |
481 | return (pageNum == 1 || pageNum == 2); | |
482 | } | |
483 | ||
484 | void MyPrintout::DrawPageOne() | |
485 | { | |
486 | // You might use THIS code if you were scaling graphics of known size to fit | |
487 | // on the page. The commented-out code illustrates different ways of scaling | |
488 | // the graphics. | |
489 | ||
490 | // We know the graphic is 230x350. If we didn't know this, we'd need to | |
491 | // calculate it. | |
492 | wxCoord maxX = 230; | |
493 | wxCoord maxY = 350; | |
494 | ||
495 | // This sets the user scale and origin of the DC so that the image fits | |
496 | // within the paper rectangle (but the edges could be cut off by printers | |
497 | // that can't print to the edges of the paper -- which is most of them. Use | |
498 | // this if your image already has its own margins. | |
499 | // FitThisSizeToPaper(wxSize(maxX, maxY)); | |
500 | // wxRect fitRect = GetLogicalPaperRect(); | |
501 | ||
502 | // This sets the user scale and origin of the DC so that the image fits | |
503 | // within the page rectangle, which is the printable area on Mac and MSW | |
504 | // and is the entire page on other platforms. | |
505 | // FitThisSizeToPage(wxSize(maxX, maxY)); | |
506 | // wxRect fitRect = GetLogicalPageRect(); | |
507 | ||
508 | // This sets the user scale and origin of the DC so that the image fits | |
509 | // within the page margins as specified by g_PageSetupData, which you can | |
510 | // change (on some platforms, at least) in the Page Setup dialog. Note that | |
511 | // on Mac, the native Page Setup dialog doesn't let you change the margins | |
512 | // of a wxPageSetupDialogData object, so you'll have to write your own dialog or | |
513 | // use the Mac-only wxMacPageMarginsDialog, as we do in this program. | |
514 | FitThisSizeToPageMargins(wxSize(maxX, maxY), *g_pageSetupData); | |
515 | wxRect fitRect = GetLogicalPageMarginsRect(*g_pageSetupData); | |
516 | ||
517 | // This sets the user scale and origin of the DC so that the image appears | |
518 | // on the paper at the same size that it appears on screen (i.e., 10-point | |
519 | // type on screen is 10-point on the printed page) and is positioned in the | |
520 | // top left corner of the page rectangle (just as the screen image appears | |
521 | // in the top left corner of the window). | |
522 | // MapScreenSizeToPage(); | |
523 | // wxRect fitRect = GetLogicalPageRect(); | |
524 | ||
525 | // You could also map the screen image to the entire paper at the same size | |
526 | // as it appears on screen. | |
527 | // MapScreenSizeToPaper(); | |
528 | // wxRect fitRect = GetLogicalPaperRect(); | |
529 | ||
530 | // You might also wish to do you own scaling in order to draw objects at | |
531 | // full native device resolution. In this case, you should do the following. | |
532 | // Note that you can use the GetLogicalXXXRect() commands to obtain the | |
533 | // appropriate rect to scale to. | |
534 | // MapScreenSizeToDevice(); | |
535 | // wxRect fitRect = GetLogicalPageRect(); | |
536 | ||
537 | // Each of the preceding Fit or Map routines positions the origin so that | |
538 | // the drawn image is positioned at the top left corner of the reference | |
539 | // rectangle. You can easily center or right- or bottom-justify the image as | |
540 | // follows. | |
541 | ||
542 | // This offsets the image so that it is centered within the reference | |
543 | // rectangle defined above. | |
544 | wxCoord xoff = (fitRect.width - maxX) / 2; | |
545 | wxCoord yoff = (fitRect.height - maxY) / 2; | |
546 | OffsetLogicalOrigin(xoff, yoff); | |
547 | ||
548 | // This offsets the image so that it is positioned at the bottom right of | |
549 | // the reference rectangle defined above. | |
550 | // wxCoord xoff = (fitRect.width - maxX); | |
551 | // wxCoord yoff = (fitRect.height - maxY); | |
552 | // OffsetLogicalOrigin(xoff, yoff); | |
553 | ||
554 | frame->Draw(*GetDC()); | |
555 | } | |
556 | ||
557 | void MyPrintout::DrawPageTwo() | |
558 | { | |
559 | // You might use THIS code to set the printer DC to ROUGHLY reflect | |
560 | // the screen text size. This page also draws lines of actual length | |
561 | // 5cm on the page. | |
562 | ||
563 | // Compare this to DrawPageOne(), which uses the really convenient routines | |
564 | // from wxPrintout to fit the screen image onto the printed page. This page | |
565 | // illustrates how to do all the scaling calculations yourself, if you're so | |
566 | // inclined. | |
567 | ||
568 | wxDC *dc = GetDC(); | |
569 | ||
570 | // Get the logical pixels per inch of screen and printer | |
571 | int ppiScreenX, ppiScreenY; | |
572 | GetPPIScreen(&ppiScreenX, &ppiScreenY); | |
573 | int ppiPrinterX, ppiPrinterY; | |
574 | GetPPIPrinter(&ppiPrinterX, &ppiPrinterY); | |
575 | ||
576 | // This scales the DC so that the printout roughly represents the the screen | |
577 | // scaling. The text point size _should_ be the right size but in fact is | |
578 | // too small for some reason. This is a detail that will need to be | |
579 | // addressed at some point but can be fudged for the moment. | |
580 | float scale = (float)((float)ppiPrinterX/(float)ppiScreenX); | |
581 | ||
582 | // Now we have to check in case our real page size is reduced (e.g. because | |
583 | // we're drawing to a print preview memory DC) | |
584 | int pageWidth, pageHeight; | |
585 | int w, h; | |
586 | dc->GetSize(&w, &h); | |
587 | GetPageSizePixels(&pageWidth, &pageHeight); | |
588 | ||
589 | // If printer pageWidth == current DC width, then this doesn't change. But w | |
590 | // might be the preview bitmap width, so scale down. | |
591 | float overallScale = scale * (float)(w/(float)pageWidth); | |
592 | dc->SetUserScale(overallScale, overallScale); | |
593 | ||
594 | // Calculate conversion factor for converting millimetres into logical | |
595 | // units. There are approx. 25.4 mm to the inch. There are ppi device units | |
596 | // to the inch. Therefore 1 mm corresponds to ppi/25.4 device units. We also | |
597 | // divide by the screen-to-printer scaling factor, because we need to | |
598 | // unscale to pass logical units to DrawLine. | |
599 | ||
600 | // Draw 50 mm by 50 mm L shape | |
601 | float logUnitsFactor = (float)(ppiPrinterX/(scale*25.4)); | |
602 | float logUnits = (float)(50*logUnitsFactor); | |
603 | dc->SetPen(* wxBLACK_PEN); | |
604 | dc->DrawLine(50, 250, (long)(50.0 + logUnits), 250); | |
605 | dc->DrawLine(50, 250, 50, (long)(250.0 + logUnits)); | |
606 | ||
607 | dc->SetBackgroundMode(wxTRANSPARENT); | |
608 | dc->SetBrush(*wxTRANSPARENT_BRUSH); | |
609 | ||
610 | { // GetTextExtent demo: | |
611 | wxString words[7] = {_T("This "), _T("is "), _T("GetTextExtent "), _T("testing "), _T("string. "), _T("Enjoy "), _T("it!")}; | |
612 | long w, h; | |
613 | long x = 200, y= 250; | |
614 | wxFont fnt(15, wxSWISS, wxNORMAL, wxNORMAL); | |
615 | ||
616 | dc->SetFont(fnt); | |
617 | ||
618 | for (int i = 0; i < 7; i++) | |
619 | { | |
620 | wxString word = words[i]; | |
621 | word.Remove( word.Len()-1, 1 ); | |
622 | dc->GetTextExtent(word, &w, &h); | |
623 | dc->DrawRectangle(x, y, w, h); | |
624 | dc->GetTextExtent(words[i], &w, &h); | |
625 | dc->DrawText(words[i], x, y); | |
626 | x += w; | |
627 | } | |
628 | ||
629 | } | |
630 | ||
631 | dc->SetFont(wxGetApp().m_testFont); | |
632 | ||
633 | dc->DrawText(_T("Some test text"), 200, 300 ); | |
634 | ||
635 | // TESTING | |
636 | ||
637 | int leftMargin = 20; | |
638 | int rightMargin = 20; | |
639 | int topMargin = 20; | |
640 | int bottomMargin = 20; | |
641 | ||
642 | int pageWidthMM, pageHeightMM; | |
643 | GetPageSizeMM(&pageWidthMM, &pageHeightMM); | |
644 | ||
645 | float leftMarginLogical = (float)(logUnitsFactor*leftMargin); | |
646 | float topMarginLogical = (float)(logUnitsFactor*topMargin); | |
647 | float bottomMarginLogical = (float)(logUnitsFactor*(pageHeightMM - bottomMargin)); | |
648 | float rightMarginLogical = (float)(logUnitsFactor*(pageWidthMM - rightMargin)); | |
649 | ||
650 | dc->SetPen(* wxRED_PEN); | |
651 | dc->DrawLine( (long)leftMarginLogical, (long)topMarginLogical, | |
652 | (long)rightMarginLogical, (long)topMarginLogical); | |
653 | dc->DrawLine( (long)leftMarginLogical, (long)bottomMarginLogical, | |
654 | (long)rightMarginLogical, (long)bottomMarginLogical); | |
655 | ||
656 | WritePageHeader(this, dc, _T("A header"), logUnitsFactor); | |
657 | } | |
658 | ||
659 | // Writes a header on a page. Margin units are in millimetres. | |
660 | bool WritePageHeader(wxPrintout *printout, wxDC *dc, const wxChar *text, float mmToLogical) | |
661 | { | |
662 | /* | |
663 | static wxFont *headerFont = (wxFont *) NULL; | |
664 | if (!headerFont) | |
665 | { | |
666 | headerFont = wxTheFontList->FindOrCreateFont(16, wxSWISS, wxNORMAL, wxBOLD); | |
667 | } | |
668 | dc->SetFont(headerFont); | |
669 | */ | |
670 | ||
671 | int pageWidthMM, pageHeightMM; | |
672 | ||
673 | printout->GetPageSizeMM(&pageWidthMM, &pageHeightMM); | |
674 | wxUnusedVar(pageHeightMM); | |
675 | ||
676 | int leftMargin = 10; | |
677 | int topMargin = 10; | |
678 | int rightMargin = 10; | |
679 | ||
680 | float leftMarginLogical = (float)(mmToLogical*leftMargin); | |
681 | float topMarginLogical = (float)(mmToLogical*topMargin); | |
682 | float rightMarginLogical = (float)(mmToLogical*(pageWidthMM - rightMargin)); | |
683 | ||
684 | long xExtent, yExtent; | |
685 | dc->GetTextExtent(text, &xExtent, &yExtent); | |
686 | float xPos = (float)(((((pageWidthMM - leftMargin - rightMargin)/2.0)+leftMargin)*mmToLogical) - (xExtent/2.0)); | |
687 | dc->DrawText(text, (long)xPos, (long)topMarginLogical); | |
688 | ||
689 | dc->SetPen(* wxBLACK_PEN); | |
690 | dc->DrawLine( (long)leftMarginLogical, (long)(topMarginLogical+yExtent), | |
691 | (long)rightMarginLogical, (long)topMarginLogical+yExtent ); | |
692 | ||
693 | return true; | |
694 | } |