franksunny的个人技术空间
获得人生中的成功需要的专注与坚持不懈多过天才与机会。 ——C.W. Wendte

MFC 打印程序的编制

MFC 在实现打印编程工作时已经建立了一个已有的框架,我们只要往这个框架里面填东西即可。

打印工作其实就是在 DC 上绘图,不过这里的 DC 是打印机的 DC ,明确这一点之后,想当然的,我们打印的任务可以分为获取打印机 DC 、根据打印机驱动让用户交互设置打印参数、开始打印、打印的具体过程(分页、绘制等)、结束打印、释放 DC 。以下通过一段示列代码来说明:

// 打印控制函数

void print()

{

       CPrintDialog dlg(FALSE,PD_ALLPAGES);// 如为 TRUE 则不能实现打印功能, // 而是建立打印机,祥查 msdn

       // Get the default printer.

       dlg.m_pd.nMinPage = 1;

       dlg.m_pd.nMaxPage = 15;

       dlg.m_pd.nCopies = 1;

       dlg.m_pd.nFromPage = 1;

       dlg.m_pd.nToPage = 2;

       if (dlg.DoModal() == IDOK)

       {// 如果仅打印一份则可以用 dlg.GetDefaults();

              HDC hdcPrinter = dlg.GetPrinterDC();

              if (hdcPrinter == NULL) // Is a default printer set up?

              {

                     AfxMessageBox(_T("Buy a printer!"));

              }

              else

              {

                     CDC dcPrinter;

                     dcPrinter.Attach(hdcPrinter);

                     // 通知打印机驱动程序接受打印文档

                     DOCINFO docinfo;

                     memset(&docinfo, 0, sizeof(docinfo));

                     docinfo.cbSize = sizeof(docinfo);

                     docinfo.lpszDocName = _T("CDC::StartDoc() Code Fragment");

                     DEVMODE * dm = dlg.GetDevMode();

                     float rato = dm->dmPrintQuality / 25.4;// 转化到 mm 模式下

                     // If it fails, complain and exit gracefully.

                     if (dcPrinter.StartDoc(&docinfo) < 0)

                     {

                            AfxMessageBox(_T("Printer would not initalize"));

                     }

                     else

                     {

                            // Do printing.

                            BOOL bPrintOK = TRUE;

                            int m_nCopies = dlg.m_pd.nCopies;

                            int page_Min(0),page_Max(0);

                            if (m_nCopies > 1)

                            {

                                   // 是否逐份打印

                                   if (dlg.PrintCollate() == TRUE)

                                   {

                                          // 是否全部打印

                                          if (dlg.PrintAll() == TRUE)

                                          {

                                                 // 全部打印

                                                 page_Min = dlg.m_pd.nMinPage;

                                                 page_Max = dlg.m_pd.nMaxPage;

                                          }

                                          else if (dlg.PrintRange() == TRUE)

                                          {

                                                 // 选定页码范围

                                                 page_Min = dlg.m_pd.nFromPage;

                                                 page_Max = dlg.m_pd.nToPage;

                                          }

                                          else if (dlg.PrintSelection() == TRUE)

                                          {

                                                 // 选定的选择区域

                                                 // 暂时不知道怎么处理的

                                          }

                                          for(int i = 0; i < m_nCopies; i++)

                                          {

                                                 for(UINT j = page_Min; j < page_Max + 1; j++)

                                                 {

                                                        if (dcPrinter.StartPage() < 0)// 只起到判断作用,不 // 加不会影响正常打印

                                                        {

                                                               AfxMessageBox(_T("Could not start page"));

                                                               bPrintOK = FALSE;

                                                               break;

                                                        }                                

                                                        OnPrint(dcPrinter, j, rato);

                                                        if (dcPrinter.EndPage() <= 0)  // 告诉打印驱动换 // 页,如果不加则将所有页打在一张纸上

                                                        {

                                                               AfxMessageBox(_T("Could not end page"));

                                                               bPrintOK = FALSE;

                                                               break;

                                                        }

                                                 }

                                                 if (bPrintOK == FALSE)

                                                 {

                                                        dcPrinter.AbortDoc();

                                                        break;

                                                 }

                                          }

                                   }

                                   else

                                   {

                                          // 是否全部打印

                                          if (dlg.PrintAll() == TRUE)

                                          {

                                                 // 全部打印

                                                 page_Min = dlg.m_pd.nMinPage;

                                                 page_Max = dlg.m_pd.nMaxPage;

                                          }

                                          else if (dlg.PrintRange() == TRUE)

                                          {

                                                 // 选定页码范围

                                                 page_Min = dlg.m_pd.nFromPage;

                                                 page_Max = dlg.m_pd.nToPage;

                                          }

                                          else if (dlg.PrintSelection() == TRUE)

                                          {

                                                 // 选定的选择区域

                                                 // 暂时不知道怎么处理的

                                          }

                                          for(UINT j = page_Min; j < page_Max + 1; j++)

                                          {

                                                 for(int i = 0; i < m_nCopies; i++)

                                                 {

                                                        if (dcPrinter.StartPage() < 0)// 只起到判断作用,不 // 加不会影响正常打印

                                                        {

                                                               AfxMessageBox(_T("Could not start page"));

                                                               bPrintOK = FALSE;

                                                               break;

                                                        }                                

                                                        OnPrint(dcPrinter, j, rato);

                                                        if (dcPrinter.EndPage() <= 0)  // 告诉打印驱动换 // 页,如果不加则将所有页打在一张纸上

                                                        {

                                                               AfxMessageBox(_T("Could not end page"));

                                                               bPrintOK = FALSE;

                                                               break;

                                                        }

                                                 }

                                                 if (bPrintOK == FALSE)

                                                 {

                                                        dcPrinter.AbortDoc();

                                                        break;

                                                 }

                                          }

                                   }

                            }

                            else

                            {

                                   // 是否全部打印

                                   if (dlg.PrintAll() == TRUE)

                                   {

                                          // 全部打印

                                          page_Min = dlg.m_pd.nMinPage;

                                          page_Max = dlg.m_pd.nMaxPage;

                                   }

                                   else if (dlg.PrintRange() == TRUE)

                                   {

                                          // 选定页码范围

                                          page_Min = dlg.m_pd.nFromPage;

                                          page_Max = dlg.m_pd.nToPage;

                                   }

                                   else if (dlg.PrintSelection() == TRUE)

                                   {

                                          // 选定的选择区域

                                          // 暂时不知道怎么处理的

                                   }

                                   for(UINT i = page_Min; i < page_Max + 1; i++)

                                   {

                                          if (dcPrinter.StartPage() < 0)// 只起到判断作用,不加不会 // 影响正常打印

                                          {

                                                 AfxMessageBox(_T("Could not start page"));

                                                 bPrintOK = FALSE;

dcPrinter.AbortDoc();

                                                 break;

                                          }                                

                                          OnPrint(dcPrinter, i, rato);

                                          if (dcPrinter.EndPage() <= 0)// 告诉打印驱动换页,如果不 // 加则将所有页打在一张纸上

                                          {

                                                 AfxMessageBox(_T("Could not end page"));

                                                 bPrintOK = FALSE;

dcPrinter.AbortDoc();

                                                 break;

                                          }

                                   }

                            }

                            if (bPrintOK == TRUE)

                            {

                                   dcPrinter.EndDoc();// 结束打印

                            }

                            dcPrinter.Detach();// 释放 DC

                     }           

              }

       }    

}

 

// 具体的页面绘制函数

void OnPrint(CDC &dc, UINT &nCurPage, float &rato)

{

       CFont font;

       font.CreatePointFont(100 * rato, " 宋体 ");

       CGdiObject* pOldFont = dc.SelectObject(&font);

       CString str;

       str.Format("Hello World! -- %d",nCurPage);

       dc.TextOut(50 * rato, 50 * rato, str);

       dc.SelectObject(pOldFont);

}

 

以上打印过程没有考虑打印预览功能,只是实现直接打印功能,现在的虚拟打印机软件( pdffactory )已经很多了,个人感觉没必要打印预览功能。如果要要想实现打印预览功能,那么我们就要考虑借用 MFC 的文档-视图结构或者对话框结构,原理是一样的。介绍此类的文档网络上也很多,我刚开始是看 http://www.vckbase.com/document/viewdoc/?id=1618 才打算写上面这么一篇的。后来发现 http://www.vckbase.com/document/viewdoc/?id=1270 ,个人觉得很值得一读的。

 

以下针对微软封装的几个相应函数按调用顺序做一简单描述:

1.OnPreparePrinting

OnPreparePrinting 函数最先被调用,用来初始化打印机等。比如,若没有安装打印机,则该函数将提示用户安装打印机。用户程序可以向其中加入别的初始化代码,比如,计算打印你的文档所需要的总页数,然后调用视类中的打印机初始化函数BOOL   DoPreparePrinting(CPrintInfopInfo)即可。而用AppWizard生成的代码中,OnPreparePrinting函数将只是调用函数DoPreparePrinting,并传递参数。

2.OnBeingPrinting

OnBeingPrinting 函数是开始打印文档前调用的函数,用户可以在其中加入另一些对于打印过程的初始化代码,比如分配打印过程中将要使用的CPen)、刷子CBrush)等,默认的代码中该函数将直接返回。

3.OnPrepareDC

OnPrepareDC 函数用于在打印前准备打印设备场,如窗口大小、原点,视图大小、原点等。同时该函数在视类显示文档内容时也被调用,默认的代码中该函数调用基类中的OnPrepareDC函数。

4.OnPrint

OnPrint 函数则是具体的打印过程,它利用前面准备好的设备场进行打印。

5.OnEndPrinting

OnEndPrinting 函数是与OnBeginPrinting函数相对应的函数,它在打印完成后由应用框架调用,用于释放在OnBeginPrinting中分配的对象,如刷子等,其默认的代码中该函数将直接返回。

 

这几个函数中,OnPreparePrintingOnBeginPrintingOnEndPrinting函数在一次打印过程中将只被调用一次,不管这一次打印内容有多少页,而OnPrepareDCOnPrint函数则每打印一页都将被调用一次,这种调用次序对于打印是很有用的。

 

好了,关于打印就总结到这里,最后还推荐一篇文章,对打印总括性比较好,如果觉得看我的代码太累就看看http://www.vckbase.com/document/viewdoc/?id=1041

 

 

posted on 2007-01-26 22:51 frank.sunny 阅读(6106) 评论(3)  编辑 收藏 引用 所属分类: 编程实践小结(MFC、C++)

FeedBack:
# re: MFC打印程序的编制
2008-01-02 15:26 | wei
太感谢了,嘻嘻  回复  更多评论
  
# re: MFC打印程序的编制[未登录]
2012-05-24 14:12 | Tom
干嘛要用绿色的背景啊,眼睛都看花了。哎。  回复  更多评论
  
# re: MFC打印程序的编制
2015-10-12 00:46 | 匿名
我为什么打印出来的是一张白纸??  回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理



常用链接

留言簿(13)

随笔分类

个人其它博客

基础知识链接

最新评论

阅读排行榜

评论排行榜