上一页 下一页 返回

8.6 打印和打印预览

  最后,我们要给绘图程序增加打印和打印预览功能。我们希望文档分两页打印,第一页为封面,打印文档名字。第二页输出文档内容,并在页眉上打印文档名字。虽然AppWizard已经自动生成了打印和打印预览的代码,但是许多情况下,并不能符合要求。

这是因为:

1.打印机和窗口(屏幕)显示的分辨率不同:打印机的分辨率用每英寸多少个点来描述,屏幕分辨率用单位面积的像素点来表示。对于同样的Arial字体下的一个字符,在屏幕上用20个像素表示,而在打印机上则需要50点。在编辑器程序中,使用的映射模式为MM_TEXT,在这种模式下,一个逻辑单位对应于一个像素点。我们已经知道,Windows是按照逻辑单位来绘图的。这样,根据MM_TEXT模式的逻辑单位(实际上也就是像素数目)决定比例的原则打印出来得内容自然要比屏幕上看到的要小的多。因此,前面在初始化视图OnInitialUpdate时候,在选择绘图的映射模式上,没有采用以前使用的缺省的MM_TEXT模式,而是采用了MM_LOENGLISH。

2.窗口和打印机对边界的处理不同:窗口可以看作是无边界的,可以在窗口外面画,而不会引起错误,窗口会自动剪裁超出边界的图形。但打印机却不同,它是按页打印的。打印输出时必须自己处理分页和换页,如果不作这样的处理的话,行和行之间就会叠加起来。

  要正确打印输出屏幕上的内容,就必须解决以上两个问题。对于第一个问题,有两种方法:一是利用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中扣除页脚的高度。