天天看點

MFC列印機制

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"));

                                                        }

                                                 }

                                                 if (bPrintOK == FALSE)

                                                        dcPrinter.AbortDoc();

                                                        break;

                                   }

                                   else

                                                 // 標明頁碼範圍

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

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

                                                        OnPrint(dcPrinter, j, rato);

                                                 {

                            }

                            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"));

                            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);

以下針對微軟封裝的幾個相應函數按調用順序做一簡單描述:

1.OnPreparePrinting

OnPreparePrinting 函數最先被調用,用來初始化列印機等。比如,若沒有安裝列印機,則該函數将提示使用者安裝列印機。使用者程式可以向其中加入别的初始化代碼,比如,計算列印你的文檔所需要的總頁數,然後調用視類中的列印機初始化函數BOOL   DoPreparePrinting(CPrintInfopInfo)即可。而用AppWizard生成的代碼中,OnPreparePrinting函數将隻是調用函數DoPreparePrinting,并傳遞參數。

2.OnBeingPrinting

OnBeingPrinting 函數是開始列印文檔前調用的函數,使用者可以在其中加入另一些對于列印過程的初始化代碼,比如配置設定列印過程中将要使用的“筆”(CPen)、“刷子”(CBrush)等,預設的代碼中該函數将直接傳回。

3.OnPrepareDC

OnPrepareDC 函數用于在列印前準備列印裝置場,如視窗大小、原點,視圖大小、原點等。同時該函數在視類顯示文檔内容時也被調用,預設的代碼中該函數調用基類中的OnPrepareDC函數。

4.OnPrint

OnPrint 函數則是具體的列印過程,它利用前面準備好的裝置場進行列印。

5.OnEndPrinting

OnEndPrinting 函數是與OnBeginPrinting函數相對應的函數,它在列印完成後由應用架構調用,用于釋放在OnBeginPrinting中配置設定的“對象”,如“筆”、“刷子”等,其預設的代碼中該函數将直接傳回。

 SetViewportOrg(0,originy);    實作獎 CView OnDraw内容分頁。

 CPrintPreviewState* pState = new CPrintPreviewState;
 CWnd* pMainWnd = GetParentFrame();
 CWinThread *pThread = AfxGetThread();
 pThread->m_pMainWnd = pMainWnd;
 pThread->m_pActiveWnd = pMainWnd;
 CFrameWnd *pParent = (CFrameWnd*)pMainWnd;
 if (DYNAMIC_DOWNCAST(CFrameWnd, pMainWnd) == NULL)
 {
 }
 // Create the preview view object
 CPrintPreView* pView = (CPrintPreView*)RUNTIME_CLASS(CPrintPreView)->CreateObject();
 if (pView == NULL)
 {
  return ;
 }
 if(pView->SetConfigCWnd(pParent,this,AFX_IDD_PREVIEW_TOOLBAR,pState))
 {
  delete pView;
  delete pState;
  return ;
 }
 if (!pView->SetPrintView(this))
 {
  delete pView;
  delete pState;
  return ;            // signal that OnEndPrintPreview was called
 }
 pParent->SetActiveView(pView);  // set active view - even for MDI
 pParent->RecalcLayout();            // position and size everything
 pParent->UpdateWindow();/*
 CPrintPreviewState* pState = new CPrintPreviewState;
 CWinThread *pThread = AfxGetThread();
 pThread->m_pMainWnd = this->GetParent(); 
 // pThread->m_pMainWnd = this->GetParentFrame(); 
 TRY
 {
  if (!DoPrintPreview(AFX_IDD_PREVIEW_TOOLBAR, this,
   RUNTIME_CLASS(CPrintPreView), pState))
  {
   AfxMessageBox(AFX_IDP_COMMAND_FAILURE);
   delete pState;     
  }
 }
 CATCH_ALL(e)
 {
  delete pState;
  THROW_LAST();
 }
 END_CATCH_ALL*/