|
Goto Index
8.6 打印和打印预览 最后,我们要给绘图程序增加打印和打印预览功能。我们希望文档分两页打印,第一页为封面,打印文档名字。第二页输出文档内容,并在页眉上打印文档名字。虽然AppWizard已经自动生成了打印和打印预览的代码,但是许多情况下,并不能符合要求。
要正确打印输出屏幕上的内容,就必须解决以上两个问题。对于第一个问题,有两种方法:一是利用SetMapMode(int nMode) 设置别的映射模式,比如采用MM_LOENGLISH,不用像素而是采用0.01inch来衡量。 要处理打印分页、换页,就必须修改框架处理打印消息的缺省行为,在其中计算和换页。此外,我们还希望在打印时在页眉处能够输出标题(使用文件名作为标题)、在页脚处输出页码。 为了实现打印和打印预览功能,首先需要了解MFC的打印体系结构,即框架是如何处理打印文档的要求的。 MFC的打印工作大致上是这样进行的: 1.显示Print对话框 2.创建一个与当前打印机设置相匹配的设备上下文(CDC)对象。 3.设置要打印的页数 4.调用CDC::StartDoc开始打印 5.用CDC::StartPage开始打印一页 6.调用视图的OnDraw()方法打印输出一页内容 7.用CDC::EndPage结束一页的打印 8.循环输出全部内容 9.用CDC::EndDoc结束打印 10.视图作打印的清理工作 框架的打印文档功能是从OnPreparePrinting(CPrintInfo* pInfo)开始的,在缺省的情况下,它只是简单的调用视图的DoPreparePrinting()函数。DoPreparePrinting()显示Print对话框,并创建与打印机相匹配的设备上下文。如果要想改变打印机初始设置,可以在这里改。缺省设置下,使用1作为第一页编号(注意:打印的页号是从1开始编号而不是0),用0xFFFF作为文档的最后一页编号。因为Draw要求分两页打印输出,因此要在这里设置打印页数。要设置打印页数,可以调用CPrintInfo::SetMaxPage(nMaxPage)。同时还将预览页数也设置为两页。 BOOL CDrawView::OnPreparePrinting(CPrintInfo* pInfo) {
pInfo->SetMaxPage(2); // the document is two pages long: // the first page is the title page // the second is the drawing BOOL bRet = DoPreparePrinting(pInfo); // default preparation pInfo->m_nNumPreviewPages = 2; // Preview 2 pages at a time // Set this value after calling DoPreparePrinting to override // value read from .INI file return bRet; } DoPreparePrinting显示Print对话框。返回时,CPrintInfo结构包含了用户所指定的值,包括起止页号、最大页号、最小页号等。 OnBeginPrinting()在OnPreparePrinting()被调用之后实际打印之前调用。OnBeginPrinting()用于分配GDI资源,这里使用缺省行为。 OnPrepareDC用作屏幕显示时,在绘图前调整DC。在用于打印时,OnPrepareDC也完成类似功能。 OnPrint完成真正的打印一页文档的工作。它把一个打印机设备上下文传给OnDraw,由OnDraw负责打印输出。可以把那些适合于打印但是不适合于屏幕输出的工作,如打印页眉和页脚,放在OnPrint()的重载中完成,然后再调用OnDraw完成打印和显示都需要的工作。现在,我们就在OnPrint中加入打印页眉和页脚的代码。OnPrint不是由AppWizard自动生成的,首先要用ClassWizard为CDrawView增加OnPrint()方法。然后添加绘图程序的特殊打印代码,见清单8.10。
清单8.10 OnPrint()成员函数 void CDrawView::OnPrint(CDC* pDC, CPrintInfo* pInfo) {
// TODO: Add your specialized code here and/or call the base class if (pInfo->m_nCurPage == 1) // page no. 1 is the title page { PrintTitlePage(pDC, pInfo); return; // nothing else to print on page 1 but the page title } CString strHeader = GetDocument()->GetTitle(); PrintPageHeader(pDC, pInfo, strHeader); // PrintPageHeader() subtracts out from the pInfo->m_rectDraw the // amount of the page used for the header. pDC->SetWindowOrg(pInfo->m_rectDraw.left,-pInfo->m_rectDraw.top); // Now print the rest of the page OnDraw(pDC); } OnPrint()首先根据CPrintInfo类型的pInfo中m_nCurPage(保存当前打印页号信息)判断当前打印的是不是第一页。如果是第一页,就打印输出封面。否则,首先调用PrintPageHeader打印页眉。然后用SetWindowOrg调整打印输出原点位置。m_rectDraw又是CPrintInfo结构的一个重要数据成员,它保存的是打印输出的矩形边界。最后将与打印机匹配的设备上下文传给OnDraw,由OnDraw在打印机上输出。注意这里使用的映射模式为MM_LOENGLISH,它的y轴方向是向上递增的。 PrintTitlePage打印输出文档的封面。它首先定义一种逻辑字体,设置逻辑字体属性,然后由调用CreateFontIndirect由逻辑字体创建字体。SetTextAlign(TA_CENTER)将文本设置为居中输出。然后调用TextOut在打印矩形m_rectDraw上输出封面。PrintTitlePage函数定义见清单8.11。
清单8.11 PrintTitlePage成员函数 void CDrawView::PrintTitlePage(CDC* pDC, CPrintInfo* pInfo) { // Prepare a font size for displaying the file name LOGFONT logFont; memset(&logFont, 0, sizeof(LOGFONT)); logFont.lfHeight = 75; // 3/4th inch high in MM_LOENGLISH // (1/100th inch) CFont font; CFont* pOldFont = NULL; if (font.CreateFontIndirect(&logFont)) pOldFont = pDC->SelectObject(&font); // Get the file name, to be displayed on title page CString strPageTitle = GetDocument()->GetTitle(); // Display the file name 1 inch below top of the page, // centered horizontally pDC->SetTextAlign(TA_CENTER); pDC->TextOut(pInfo->m_rectDraw.right/2, -100, strPageTitle); if (pOldFont != NULL) pDC->SelectObject(pOldFont); } PrintPageHeader在页眉位置输出文件名,然后从m_rectDraw扣除页眉的大小。
void CDrawView::PrintPageHeader(CDC* pDC, CPrintInfo* pInfo, CString& strHeader) { // Print a page header consisting of the name of // the document and a horizontal line pDC->SetTextAlign(TA_LEFT); pDC->TextOut(0,-25, strHeader); // 1/4 inch down // Draw a line across the page, below the header TEXTMETRIC textMetric; pDC->GetTextMetrics(&textMetric); int y = -35 - textMetric.tmHeight; // line 1/10th inch below text pDC->MoveTo(0, y); // from left margin pDC->LineTo(pInfo->m_rectDraw.right, y); // to right margin // Subtract out from the drawing rectange the space used by the header. y -= 25; // space 1/4 inch below (top of) line pInfo->m_rectDraw.top += y; } 作为一个练习,读者可以修改OnPrint()并增加一个PrintPageFooter()函数,在每一页的页脚处输出打印的页号。注意调用OnDraw之前,要从m_rectDraw中扣除页脚的高度。
本教程幼髡? 作者: 不详, 来源: Visual C++王朝 |