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*/