Update OpenVMS compile support
[wxWidgets.git] / src / html / htmprint.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/html/htmprint.cpp
3 // Purpose: html printing classes
4 // Author: Vaclav Slavik
5 // Created: 25/09/99
6 // RCS-ID: $Id$
7 // Copyright: (c) Vaclav Slavik, 1999
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
10
11 // For compilers that support precompilation, includes "wx/wx.h".
12 #include "wx/wxprec.h"
13
14 #ifdef __BORLANDC__
15 #pragma hdrstop
16 #endif
17
18 #if wxUSE_HTML && wxUSE_PRINTING_ARCHITECTURE && wxUSE_STREAMS
19
20 #ifndef WX_PRECOMP
21 #include "wx/log.h"
22 #include "wx/intl.h"
23 #include "wx/dc.h"
24 #include "wx/settings.h"
25 #include "wx/msgdlg.h"
26 #include "wx/module.h"
27 #include "wx/sizer.h"
28 #endif
29
30 #include "wx/print.h"
31 #include "wx/printdlg.h"
32 #include "wx/html/htmprint.h"
33 #include "wx/wxhtml.h"
34 #include "wx/wfstream.h"
35 #include "wx/infobar.h"
36
37
38 // default font size of normal text (HTML font size 0) for printing, in points:
39 #define DEFAULT_PRINT_FONT_SIZE 12
40
41
42 // CSS specification offer following guidance on dealing with pixel sizes
43 // when printing at
44 // http://www.w3.org/TR/2004/CR-CSS21-20040225/syndata.html#length-units:
45 //
46 // Pixel units are relative to the resolution of the viewing device, i.e.,
47 // most often a computer display. If the pixel density of the output
48 // device is very different from that of a typical computer display, the
49 // user agent should rescale pixel values. It is recommended that the [
50 // reference pixel] be the visual angle of one pixel on a device with a
51 // pixel density of 96dpi and a distance from the reader of an arm's
52 // length. For a nominal arm's length of 28 inches, the visual angle is
53 // therefore about 0.0213 degrees.
54 //
55 // For reading at arm's length, 1px thus corresponds to about 0.26 mm
56 // (1/96 inch). When printed on a laser printer, meant for reading at a
57 // little less than arm's length (55 cm, 21 inches), 1px is about 0.20 mm.
58 // On a 300 dots-per-inch (dpi) printer, that may be rounded up to 3 dots
59 // (0.25 mm); on a 600 dpi printer, it can be rounded to 5 dots.
60 //
61 // See also http://trac.wxwidgets.org/ticket/10942.
62 #define TYPICAL_SCREEN_DPI 96.0
63
64 //--------------------------------------------------------------------------------
65 // wxHtmlDCRenderer
66 //--------------------------------------------------------------------------------
67
68
69 wxHtmlDCRenderer::wxHtmlDCRenderer() : wxObject()
70 {
71 m_DC = NULL;
72 m_Width = m_Height = 0;
73 m_Cells = NULL;
74 m_Parser = new wxHtmlWinParser();
75 m_FS = new wxFileSystem();
76 m_Parser->SetFS(m_FS);
77 SetStandardFonts(DEFAULT_PRINT_FONT_SIZE);
78 }
79
80
81
82 wxHtmlDCRenderer::~wxHtmlDCRenderer()
83 {
84 if (m_Cells) delete m_Cells;
85 if (m_Parser) delete m_Parser;
86 if (m_FS) delete m_FS;
87 }
88
89
90
91 void wxHtmlDCRenderer::SetDC(wxDC *dc, double pixel_scale, double font_scale)
92 {
93 m_DC = dc;
94 m_Parser->SetDC(m_DC, pixel_scale, font_scale);
95 }
96
97
98
99 void wxHtmlDCRenderer::SetSize(int width, int height)
100 {
101 wxCHECK_RET( width, "width must be non-zero" );
102 wxCHECK_RET( height, "height must be non-zero" );
103
104 m_Width = width;
105 m_Height = height;
106 }
107
108
109 void wxHtmlDCRenderer::SetHtmlText(const wxString& html, const wxString& basepath, bool isdir)
110 {
111 wxCHECK_RET( m_DC, "SetDC() must be called before SetHtmlText()" );
112 wxCHECK_RET( m_Width, "SetSize() must be called before SetHtmlText()" );
113
114 wxDELETE(m_Cells);
115
116 m_FS->ChangePathTo(basepath, isdir);
117 m_Cells = (wxHtmlContainerCell*) m_Parser->Parse(html);
118 m_Cells->SetIndent(0, wxHTML_INDENT_ALL, wxHTML_UNITS_PIXELS);
119 m_Cells->Layout(m_Width);
120 }
121
122
123 void wxHtmlDCRenderer::SetFonts(const wxString& normal_face, const wxString& fixed_face,
124 const int *sizes)
125 {
126 m_Parser->SetFonts(normal_face, fixed_face, sizes);
127
128 if ( m_Cells )
129 m_Cells->Layout(m_Width);
130 // else: SetHtmlText() not yet called, no need for relayout
131 }
132
133 void wxHtmlDCRenderer::SetStandardFonts(int size,
134 const wxString& normal_face,
135 const wxString& fixed_face)
136 {
137 m_Parser->SetStandardFonts(size, normal_face, fixed_face);
138
139 if ( m_Cells )
140 m_Cells->Layout(m_Width);
141 // else: SetHtmlText() not yet called, no need for relayout
142 }
143
144 int wxHtmlDCRenderer::Render(int x, int y,
145 wxArrayInt& known_pagebreaks,
146 int from, int dont_render, int to)
147 {
148 wxCHECK_MSG( m_Cells, 0, "SetHtmlText() must be called before Render()" );
149 wxCHECK_MSG( m_DC, 0, "SetDC() must be called before Render()" );
150
151 int pbreak, hght;
152
153 pbreak = (int)(from + m_Height);
154 while (m_Cells->AdjustPagebreak(&pbreak, known_pagebreaks)) {}
155 hght = pbreak - from;
156 if(to < hght)
157 hght = to;
158
159 if (!dont_render)
160 {
161 wxHtmlRenderingInfo rinfo;
162 wxDefaultHtmlRenderingStyle rstyle;
163 rinfo.SetStyle(&rstyle);
164 m_DC->SetBrush(*wxWHITE_BRUSH);
165 m_DC->SetClippingRegion(x, y, m_Width, hght);
166 m_Cells->Draw(*m_DC,
167 x, (y - from),
168 y, y + hght,
169 rinfo);
170 m_DC->DestroyClippingRegion();
171 }
172
173 if (pbreak < m_Cells->GetHeight()) return pbreak;
174 else return GetTotalHeight();
175 }
176
177 int wxHtmlDCRenderer::GetTotalWidth() const
178 {
179 return m_Cells ? m_Cells->GetWidth() : 0;
180 }
181
182 int wxHtmlDCRenderer::GetTotalHeight() const
183 {
184 return m_Cells ? m_Cells->GetHeight() : 0;
185 }
186
187
188 //--------------------------------------------------------------------------------
189 // wxHtmlPrintout
190 //--------------------------------------------------------------------------------
191
192
193 wxList wxHtmlPrintout::m_Filters;
194
195 wxHtmlPrintout::wxHtmlPrintout(const wxString& title) : wxPrintout(title)
196 {
197 m_Renderer = new wxHtmlDCRenderer;
198 m_RendererHdr = new wxHtmlDCRenderer;
199 m_NumPages = wxHTML_PRINT_MAX_PAGES;
200 m_Document = m_BasePath = wxEmptyString; m_BasePathIsDir = true;
201 m_Headers[0] = m_Headers[1] = wxEmptyString;
202 m_Footers[0] = m_Footers[1] = wxEmptyString;
203 m_HeaderHeight = m_FooterHeight = 0;
204 SetMargins(); // to default values
205 SetStandardFonts(DEFAULT_PRINT_FONT_SIZE);
206 }
207
208
209
210 wxHtmlPrintout::~wxHtmlPrintout()
211 {
212 delete m_Renderer;
213 delete m_RendererHdr;
214 }
215
216 void wxHtmlPrintout::CleanUpStatics()
217 {
218 WX_CLEAR_LIST(wxList, m_Filters);
219 }
220
221 // Adds input filter
222 void wxHtmlPrintout::AddFilter(wxHtmlFilter *filter)
223 {
224 m_Filters.Append(filter);
225 }
226
227 bool
228 wxHtmlPrintout::CheckFit(const wxSize& pageArea, const wxSize& docArea) const
229 {
230 // Nothing to do if the contents fits horizontally.
231 if ( docArea.x <= pageArea.x )
232 return true;
233
234 // Otherwise warn the user more or less intrusively depending on whether
235 // we're previewing or printing:
236 if ( wxPrintPreview * const preview = GetPreview() )
237 {
238 // Don't annoy the user too much when previewing by using info bar
239 // instead of a dialog box.
240 #if wxUSE_INFOBAR
241 wxFrame * const parent = preview->GetFrame();
242 wxCHECK_MSG( parent, false, "No parent preview frame?" );
243
244 wxSizer * const sizer = parent->GetSizer();
245 wxCHECK_MSG( sizer, false, "Preview frame should be using sizers" );
246
247 wxInfoBar * const bar = new wxInfoBar(parent);
248 sizer->Add(bar, wxSizerFlags().Expand());
249
250 // Note that the message here is similar to the one below but not
251 // exactly the same, notably we don't use the document title here
252 // because it's already clear which document it pertains to and the
253 // title may be long enough to make the text not fit in the window.
254 bar->ShowMessage
255 (
256 _("This document doesn't fit on the page horizontally and "
257 "will be truncated when it is printed."),
258 wxICON_WARNING
259 );
260 #endif // wxUSE_INFOBAR
261 }
262 else // We're going to really print and not just preview.
263 {
264 // This is our last chance to warn the user that the output will be
265 // mangled so do show a message box.
266 wxMessageDialog
267 dlg
268 (
269 NULL,
270 wxString::Format
271 (
272 _("The document \"%s\" doesn't fit on the page "
273 "horizontally and will be truncated if printed.\n"
274 "\n"
275 "Would you like to proceed with printing it nevertheless?"),
276 GetTitle()
277 ),
278 _("Printing"),
279 wxOK | wxCANCEL | wxCANCEL_DEFAULT | wxICON_QUESTION
280 );
281 dlg.SetExtendedMessage
282 (
283 _("If possible, try changing the layout parameters to "
284 "make the printout more narrow.")
285 );
286 dlg.SetOKLabel(wxID_PRINT);
287
288 if ( dlg.ShowModal() == wxID_CANCEL )
289 return false;
290 }
291
292 return true;
293 }
294
295 void wxHtmlPrintout::OnPreparePrinting()
296 {
297 int pageWidth, pageHeight, mm_w, mm_h, dc_w, dc_h;
298 float ppmm_h, ppmm_v;
299
300 GetPageSizePixels(&pageWidth, &pageHeight);
301 GetPageSizeMM(&mm_w, &mm_h);
302 ppmm_h = (float)pageWidth / mm_w;
303 ppmm_v = (float)pageHeight / mm_h;
304
305 int ppiPrinterX, ppiPrinterY;
306 GetPPIPrinter(&ppiPrinterX, &ppiPrinterY);
307 wxUnusedVar(ppiPrinterX);
308 int ppiScreenX, ppiScreenY;
309 GetPPIScreen(&ppiScreenX, &ppiScreenY);
310 wxUnusedVar(ppiScreenX);
311
312 GetDC()->GetSize(&dc_w, &dc_h);
313
314 GetDC()->SetUserScale((double)dc_w / (double)pageWidth,
315 (double)dc_h / (double)pageHeight);
316
317 /* prepare headers/footers renderer: */
318
319 m_RendererHdr->SetDC(GetDC(),
320 (double)ppiPrinterY / TYPICAL_SCREEN_DPI,
321 (double)ppiPrinterY / (double)ppiScreenY);
322 m_RendererHdr->SetSize((int) (ppmm_h * (mm_w - m_MarginLeft - m_MarginRight)),
323 (int) (ppmm_v * (mm_h - m_MarginTop - m_MarginBottom)));
324 if (m_Headers[0] != wxEmptyString)
325 {
326 m_RendererHdr->SetHtmlText(TranslateHeader(m_Headers[0], 1));
327 m_HeaderHeight = m_RendererHdr->GetTotalHeight();
328 }
329 else if (m_Headers[1] != wxEmptyString)
330 {
331 m_RendererHdr->SetHtmlText(TranslateHeader(m_Headers[1], 1));
332 m_HeaderHeight = m_RendererHdr->GetTotalHeight();
333 }
334 if (m_Footers[0] != wxEmptyString)
335 {
336 m_RendererHdr->SetHtmlText(TranslateHeader(m_Footers[0], 1));
337 m_FooterHeight = m_RendererHdr->GetTotalHeight();
338 }
339 else if (m_Footers[1] != wxEmptyString)
340 {
341 m_RendererHdr->SetHtmlText(TranslateHeader(m_Footers[1], 1));
342 m_FooterHeight = m_RendererHdr->GetTotalHeight();
343 }
344
345 /* prepare main renderer: */
346 m_Renderer->SetDC(GetDC(),
347 (double)ppiPrinterY / TYPICAL_SCREEN_DPI,
348 (double)ppiPrinterY / (double)ppiScreenY);
349
350 const int printAreaW = int(ppmm_h * (mm_w - m_MarginLeft - m_MarginRight));
351 int printAreaH = int(ppmm_v * (mm_h - m_MarginTop - m_MarginBottom));
352 if ( m_HeaderHeight )
353 printAreaH -= int(m_HeaderHeight + m_MarginSpace * ppmm_v);
354 if ( m_FooterHeight )
355 printAreaH -= int(m_FooterHeight + m_MarginSpace * ppmm_v);
356
357 m_Renderer->SetSize(printAreaW, printAreaH);
358 m_Renderer->SetHtmlText(m_Document, m_BasePath, m_BasePathIsDir);
359
360 if ( CheckFit(wxSize(printAreaW, printAreaH),
361 wxSize(m_Renderer->GetTotalWidth(),
362 m_Renderer->GetTotalHeight())) || IsPreview() )
363 {
364 // do paginate the document
365 CountPages();
366 }
367 //else: if we don't call CountPages() m_PageBreaks remains empty and our
368 // GetPageInfo() will return 0 as max page and so nothing will be
369 // printed
370 }
371
372 bool wxHtmlPrintout::OnBeginDocument(int startPage, int endPage)
373 {
374 if (!wxPrintout::OnBeginDocument(startPage, endPage)) return false;
375
376 return true;
377 }
378
379
380 bool wxHtmlPrintout::OnPrintPage(int page)
381 {
382 wxDC *dc = GetDC();
383 if (dc && dc->IsOk())
384 {
385 if (HasPage(page))
386 RenderPage(dc, page);
387 return true;
388 }
389 else return false;
390 }
391
392
393 void wxHtmlPrintout::GetPageInfo(int *minPage, int *maxPage, int *selPageFrom, int *selPageTo)
394 {
395 *minPage = 1;
396 if ( m_NumPages >= (signed)m_PageBreaks.GetCount()-1)
397 *maxPage = m_NumPages;
398 else
399 *maxPage = (signed)m_PageBreaks.GetCount()-1;
400 *selPageFrom = 1;
401 *selPageTo = (signed)m_PageBreaks.GetCount()-1;
402 }
403
404
405
406 bool wxHtmlPrintout::HasPage(int pageNum)
407 {
408 return pageNum > 0 && (unsigned)pageNum < m_PageBreaks.GetCount();
409 }
410
411
412
413 void wxHtmlPrintout::SetHtmlText(const wxString& html, const wxString &basepath, bool isdir)
414 {
415 m_Document = html;
416 m_BasePath = basepath;
417 m_BasePathIsDir = isdir;
418 }
419
420 void wxHtmlPrintout::SetHtmlFile(const wxString& htmlfile)
421 {
422 wxFileSystem fs;
423 wxFSFile *ff;
424
425 if (wxFileExists(htmlfile))
426 ff = fs.OpenFile(wxFileSystem::FileNameToURL(htmlfile));
427 else
428 ff = fs.OpenFile(htmlfile);
429
430 if (ff == NULL)
431 {
432 wxLogError(htmlfile + _(": file does not exist!"));
433 return;
434 }
435
436 bool done = false;
437 wxHtmlFilterHTML defaultFilter;
438 wxString doc;
439
440 wxList::compatibility_iterator node = m_Filters.GetFirst();
441 while (node)
442 {
443 wxHtmlFilter *h = (wxHtmlFilter*) node->GetData();
444 if (h->CanRead(*ff))
445 {
446 doc = h->ReadFile(*ff);
447 done = true;
448 break;
449 }
450 node = node->GetNext();
451 }
452
453 if (!done)
454 doc = defaultFilter.ReadFile(*ff);
455
456 SetHtmlText(doc, htmlfile, false);
457 delete ff;
458 }
459
460
461
462 void wxHtmlPrintout::SetHeader(const wxString& header, int pg)
463 {
464 if (pg == wxPAGE_ALL || pg == wxPAGE_EVEN)
465 m_Headers[0] = header;
466 if (pg == wxPAGE_ALL || pg == wxPAGE_ODD)
467 m_Headers[1] = header;
468 }
469
470
471
472 void wxHtmlPrintout::SetFooter(const wxString& footer, int pg)
473 {
474 if (pg == wxPAGE_ALL || pg == wxPAGE_EVEN)
475 m_Footers[0] = footer;
476 if (pg == wxPAGE_ALL || pg == wxPAGE_ODD)
477 m_Footers[1] = footer;
478 }
479
480
481
482 void wxHtmlPrintout::CountPages()
483 {
484 wxBusyCursor wait;
485 int pageWidth, pageHeight, mm_w, mm_h;
486 float ppmm_h, ppmm_v;
487
488 GetPageSizePixels(&pageWidth, &pageHeight);
489 GetPageSizeMM(&mm_w, &mm_h);
490 ppmm_h = (float)pageWidth / mm_w;
491 ppmm_v = (float)pageHeight / mm_h;
492
493 int pos = 0;
494 m_NumPages = 0;
495 // m_PageBreaks[0] = 0;
496
497 m_PageBreaks.Clear();
498 m_PageBreaks.Add( 0);
499 do
500 {
501 pos = m_Renderer->Render((int)( ppmm_h * m_MarginLeft),
502 (int) (ppmm_v * (m_MarginTop + (m_HeaderHeight == 0 ? 0 : m_MarginSpace)) + m_HeaderHeight),
503 m_PageBreaks,
504 pos, true, INT_MAX);
505 m_PageBreaks.Add( pos);
506 if( m_PageBreaks.GetCount() > wxHTML_PRINT_MAX_PAGES)
507 {
508 wxMessageBox( _("HTML pagination algorithm generated more than the allowed maximum number of pages and it can't continue any longer!"),
509 _("Warning"), wxCANCEL | wxICON_ERROR );
510 break;
511 }
512 } while (pos < m_Renderer->GetTotalHeight());
513 }
514
515
516
517 void wxHtmlPrintout::RenderPage(wxDC *dc, int page)
518 {
519 wxBusyCursor wait;
520
521 int pageWidth, pageHeight, mm_w, mm_h, dc_w, dc_h;
522 float ppmm_h, ppmm_v;
523
524 GetPageSizePixels(&pageWidth, &pageHeight);
525 GetPageSizeMM(&mm_w, &mm_h);
526 ppmm_h = (float)pageWidth / mm_w;
527 ppmm_v = (float)pageHeight / mm_h;
528 dc->GetSize(&dc_w, &dc_h);
529
530 int ppiPrinterX, ppiPrinterY;
531 GetPPIPrinter(&ppiPrinterX, &ppiPrinterY);
532 wxUnusedVar(ppiPrinterX);
533 int ppiScreenX, ppiScreenY;
534 GetPPIScreen(&ppiScreenX, &ppiScreenY);
535 wxUnusedVar(ppiScreenX);
536
537 dc->SetUserScale((double)dc_w / (double)pageWidth,
538 (double)dc_h / (double)pageHeight);
539
540 m_Renderer->SetDC(dc,
541 (double)ppiPrinterY / TYPICAL_SCREEN_DPI,
542 (double)ppiPrinterY / (double)ppiScreenY);
543
544 dc->SetBackgroundMode(wxBRUSHSTYLE_TRANSPARENT);
545
546 m_Renderer->Render((int) (ppmm_h * m_MarginLeft),
547 (int) (ppmm_v * (m_MarginTop + (m_HeaderHeight == 0 ? 0 : m_MarginSpace)) + m_HeaderHeight), m_PageBreaks,
548 m_PageBreaks[page-1], false, m_PageBreaks[page]-m_PageBreaks[page-1]);
549
550
551 m_RendererHdr->SetDC(dc,
552 (double)ppiPrinterY / TYPICAL_SCREEN_DPI,
553 (double)ppiPrinterY / (double)ppiScreenY);
554 if (m_Headers[page % 2] != wxEmptyString)
555 {
556 m_RendererHdr->SetHtmlText(TranslateHeader(m_Headers[page % 2], page));
557 m_RendererHdr->Render((int) (ppmm_h * m_MarginLeft), (int) (ppmm_v * m_MarginTop), m_PageBreaks);
558 }
559 if (m_Footers[page % 2] != wxEmptyString)
560 {
561 m_RendererHdr->SetHtmlText(TranslateHeader(m_Footers[page % 2], page));
562 m_RendererHdr->Render((int) (ppmm_h * m_MarginLeft), (int) (pageHeight - ppmm_v * m_MarginBottom - m_FooterHeight), m_PageBreaks);
563 }
564 }
565
566
567
568 wxString wxHtmlPrintout::TranslateHeader(const wxString& instr, int page)
569 {
570 wxString r = instr;
571 wxString num;
572
573 num.Printf(wxT("%i"), page);
574 r.Replace(wxT("@PAGENUM@"), num);
575
576 num.Printf(wxT("%lu"), (unsigned long)(m_PageBreaks.GetCount() - 1));
577 r.Replace(wxT("@PAGESCNT@"), num);
578
579 const wxDateTime now = wxDateTime::Now();
580 r.Replace(wxT("@DATE@"), now.FormatDate());
581 r.Replace(wxT("@TIME@"), now.FormatTime());
582
583 r.Replace(wxT("@TITLE@"), GetTitle());
584
585 return r;
586 }
587
588
589
590 void wxHtmlPrintout::SetMargins(float top, float bottom, float left, float right, float spaces)
591 {
592 m_MarginTop = top;
593 m_MarginBottom = bottom;
594 m_MarginLeft = left;
595 m_MarginRight = right;
596 m_MarginSpace = spaces;
597 }
598
599
600
601
602 void wxHtmlPrintout::SetFonts(const wxString& normal_face, const wxString& fixed_face,
603 const int *sizes)
604 {
605 m_Renderer->SetFonts(normal_face, fixed_face, sizes);
606 m_RendererHdr->SetFonts(normal_face, fixed_face, sizes);
607 }
608
609 void wxHtmlPrintout::SetStandardFonts(int size,
610 const wxString& normal_face,
611 const wxString& fixed_face)
612 {
613 m_Renderer->SetStandardFonts(size, normal_face, fixed_face);
614 m_RendererHdr->SetStandardFonts(size, normal_face, fixed_face);
615 }
616
617
618
619 //----------------------------------------------------------------------------
620 // wxHtmlEasyPrinting
621 //----------------------------------------------------------------------------
622
623
624 wxHtmlEasyPrinting::wxHtmlEasyPrinting(const wxString& name, wxWindow *parentWindow)
625 {
626 m_ParentWindow = parentWindow;
627 m_Name = name;
628 m_PrintData = NULL;
629 m_PageSetupData = new wxPageSetupDialogData;
630 m_Headers[0] = m_Headers[1] = m_Footers[0] = m_Footers[1] = wxEmptyString;
631
632 m_PageSetupData->EnableMargins(true);
633 m_PageSetupData->SetMarginTopLeft(wxPoint(25, 25));
634 m_PageSetupData->SetMarginBottomRight(wxPoint(25, 25));
635
636 SetStandardFonts(DEFAULT_PRINT_FONT_SIZE);
637 }
638
639
640
641 wxHtmlEasyPrinting::~wxHtmlEasyPrinting()
642 {
643 delete m_PrintData;
644 delete m_PageSetupData;
645 }
646
647
648 wxPrintData *wxHtmlEasyPrinting::GetPrintData()
649 {
650 if (m_PrintData == NULL)
651 m_PrintData = new wxPrintData();
652 return m_PrintData;
653 }
654
655
656 bool wxHtmlEasyPrinting::PreviewFile(const wxString &htmlfile)
657 {
658 wxHtmlPrintout *p1 = CreatePrintout();
659 p1->SetHtmlFile(htmlfile);
660 wxHtmlPrintout *p2 = CreatePrintout();
661 p2->SetHtmlFile(htmlfile);
662 return DoPreview(p1, p2);
663 }
664
665
666
667 bool wxHtmlEasyPrinting::PreviewText(const wxString &htmltext, const wxString &basepath)
668 {
669 wxHtmlPrintout *p1 = CreatePrintout();
670 p1->SetHtmlText(htmltext, basepath, true);
671 wxHtmlPrintout *p2 = CreatePrintout();
672 p2->SetHtmlText(htmltext, basepath, true);
673 return DoPreview(p1, p2);
674 }
675
676
677
678 bool wxHtmlEasyPrinting::PrintFile(const wxString &htmlfile)
679 {
680 wxHtmlPrintout *p = CreatePrintout();
681 p->SetHtmlFile(htmlfile);
682 bool ret = DoPrint(p);
683 delete p;
684 return ret;
685 }
686
687
688
689 bool wxHtmlEasyPrinting::PrintText(const wxString &htmltext, const wxString &basepath)
690 {
691 wxHtmlPrintout *p = CreatePrintout();
692 p->SetHtmlText(htmltext, basepath, true);
693 bool ret = DoPrint(p);
694 delete p;
695 return ret;
696 }
697
698
699
700 bool wxHtmlEasyPrinting::DoPreview(wxHtmlPrintout *printout1, wxHtmlPrintout *printout2)
701 {
702 // Pass two printout objects: for preview, and possible printing.
703 wxPrintDialogData printDialogData(*GetPrintData());
704 wxPrintPreview *preview = new wxPrintPreview(printout1, printout2, &printDialogData);
705 if (!preview->Ok())
706 {
707 delete preview;
708 return false;
709 }
710
711 wxPreviewFrame *frame = new wxPreviewFrame(preview, m_ParentWindow,
712 m_Name + _(" Preview"),
713 wxPoint(100, 100), wxSize(650, 500));
714 frame->Centre(wxBOTH);
715 frame->Initialize();
716 frame->Show(true);
717 return true;
718 }
719
720
721
722 bool wxHtmlEasyPrinting::DoPrint(wxHtmlPrintout *printout)
723 {
724 wxPrintDialogData printDialogData(*GetPrintData());
725 wxPrinter printer(&printDialogData);
726
727 if (!printer.Print(m_ParentWindow, printout, true))
728 {
729 return false;
730 }
731
732 (*GetPrintData()) = printer.GetPrintDialogData().GetPrintData();
733 return true;
734 }
735
736
737
738
739 void wxHtmlEasyPrinting::PageSetup()
740 {
741 if (!GetPrintData()->Ok())
742 {
743 wxLogError(_("There was a problem during page setup: you may need to set a default printer."));
744 return;
745 }
746
747 m_PageSetupData->SetPrintData(*GetPrintData());
748 wxPageSetupDialog pageSetupDialog(m_ParentWindow, m_PageSetupData);
749
750 if (pageSetupDialog.ShowModal() == wxID_OK)
751 {
752 (*GetPrintData()) = pageSetupDialog.GetPageSetupData().GetPrintData();
753 (*m_PageSetupData) = pageSetupDialog.GetPageSetupData();
754 }
755 }
756
757
758
759 void wxHtmlEasyPrinting::SetHeader(const wxString& header, int pg)
760 {
761 if (pg == wxPAGE_ALL || pg == wxPAGE_EVEN)
762 m_Headers[0] = header;
763 if (pg == wxPAGE_ALL || pg == wxPAGE_ODD)
764 m_Headers[1] = header;
765 }
766
767
768
769 void wxHtmlEasyPrinting::SetFooter(const wxString& footer, int pg)
770 {
771 if (pg == wxPAGE_ALL || pg == wxPAGE_EVEN)
772 m_Footers[0] = footer;
773 if (pg == wxPAGE_ALL || pg == wxPAGE_ODD)
774 m_Footers[1] = footer;
775 }
776
777
778 void wxHtmlEasyPrinting::SetFonts(const wxString& normal_face, const wxString& fixed_face,
779 const int *sizes)
780 {
781 m_fontMode = FontMode_Explicit;
782 m_FontFaceNormal = normal_face;
783 m_FontFaceFixed = fixed_face;
784
785 if (sizes)
786 {
787 m_FontsSizes = m_FontsSizesArr;
788 for (int i = 0; i < 7; i++) m_FontsSizes[i] = sizes[i];
789 }
790 else
791 m_FontsSizes = NULL;
792 }
793
794 void wxHtmlEasyPrinting::SetStandardFonts(int size,
795 const wxString& normal_face,
796 const wxString& fixed_face)
797 {
798 m_fontMode = FontMode_Standard;
799 m_FontFaceNormal = normal_face;
800 m_FontFaceFixed = fixed_face;
801 m_FontsSizesArr[0] = size;
802 }
803
804
805 wxHtmlPrintout *wxHtmlEasyPrinting::CreatePrintout()
806 {
807 wxHtmlPrintout *p = new wxHtmlPrintout(m_Name);
808
809 if (m_fontMode == FontMode_Explicit)
810 {
811 p->SetFonts(m_FontFaceNormal, m_FontFaceFixed, m_FontsSizes);
812 }
813 else // FontMode_Standard
814 {
815 p->SetStandardFonts(m_FontsSizesArr[0],
816 m_FontFaceNormal, m_FontFaceFixed);
817 }
818
819 p->SetHeader(m_Headers[0], wxPAGE_EVEN);
820 p->SetHeader(m_Headers[1], wxPAGE_ODD);
821 p->SetFooter(m_Footers[0], wxPAGE_EVEN);
822 p->SetFooter(m_Footers[1], wxPAGE_ODD);
823
824 p->SetMargins(m_PageSetupData->GetMarginTopLeft().y,
825 m_PageSetupData->GetMarginBottomRight().y,
826 m_PageSetupData->GetMarginTopLeft().x,
827 m_PageSetupData->GetMarginBottomRight().x);
828
829 return p;
830 }
831
832 // A module to allow initialization/cleanup
833 // without calling these functions from app.cpp or from
834 // the user's application.
835
836 class wxHtmlPrintingModule: public wxModule
837 {
838 DECLARE_DYNAMIC_CLASS(wxHtmlPrintingModule)
839 public:
840 wxHtmlPrintingModule() : wxModule() {}
841 bool OnInit() { return true; }
842 void OnExit() { wxHtmlPrintout::CleanUpStatics(); }
843 };
844
845 IMPLEMENT_DYNAMIC_CLASS(wxHtmlPrintingModule, wxModule)
846
847
848 // This hack forces the linker to always link in m_* files
849 // (wxHTML doesn't work without handlers from these files)
850 #include "wx/html/forcelnk.h"
851 FORCE_WXHTML_MODULES()
852
853 #endif // wxUSE_HTML & wxUSE_PRINTING_ARCHITECTURE