MFC對話框使用CPrintDialog實現打印,指定打印機、后台打印


推薦下 不錯。

對話框打印,網上一搜一大堆,基本分2類:

A類: CPrintDialog.DoModal,然后在模態對話框里選打印機、打印配置;

B類:GetPrinterDeviceDefaults,調用默認打印機。

我的工作內容是理解以上2類后,再根據MSDN,實現MDF對話框后台指定打印機打印。

廢話不多說,上菜~

 

功能:基於對話框的MFC打印(非文檔視圖結構),指定打印機,后台打印(不彈出對話框)

 

思路:

1、枚舉打印機,並選擇其中一個;

2、CPrintDialog實例指定到選中的打印機;

3、CPrintDialog后台打印

 

具體實現:

1、變量(控件)。在對話框上添加一個combobox(IDC_COMBO1,對應變量m_cboPrint)、一個edit(IDC_EDIT1),edit允許回車,多行(代碼就不貼了,知道MFC應該就懂);

2、在OnInitDialog里枚舉打印機設備,如果報函數未定義,加入頭文件#include <winspool.h>

需要調用兩次EnumPrinters函數,第一次的到結構體的大小,第二次得到打印機列表

 

 
 
  1. // TODO: 在此添加額外的初始化代碼  
  2.     DWORD dwNeeded;  
  3.     DWORD dwReturn;  
  4.     DWORD dwFlag = PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL;  
  5.   
  6.     EnumPrinters(dwFlag, NULL, 4, NULL, 0, &dwNeeded, &dwReturn);  
  7.   
  8.     PRINTER_INFO_4* p4;  
  9.     p4 = new PRINTER_INFO_4[dwNeeded];  
  10.     EnumPrinters(dwFlag, NULL, 4, (PBYTE)p4, dwNeeded, &dwNeeded, &dwReturn);  
  11.   
  12.     for (int i = 0; i<(int)dwReturn; i++)  
  13.         this->m_cboPrint.AddString(p4[i].pPrinterName);  
  14.   
  15.     delete []p4;  



 

3、操作對話框,在IDC_EDIT1里輸入打印的內容,在IDC_COMBO1里選中打印機;

4、打印(我是用OK按鈕打印的,大家隨便)

 

[cpp]  view plain copy 在CODE上查看代碼片 派生到我的代碼片
 
 
 
  1. // TODO: 在此添加控件通知處理程序代碼  
  2. //  CDialogEx::OnOK();  
  3.     this->UpdateData();  
  4.   
  5.     CString strMessage;  
  6.     CString strPrintDevice;  
  7.   
  8.     this->GetDlgItem(IDC_EDIT1)->GetWindowTextW(strMessage);  
  9.     strMessage += _T("\r\n");       //添加結尾,方便后面循環讀取打印數據  
  10.   
  11.     this->GetDlgItem(IDC_COMBO1)->GetWindowTextW(strPrintDevice);  
  12.   
  13.     DWORD dwFlag = PD_ALLPAGES | PD_NOPAGENUMS | PD_USEDEVMODECOPIES | PD_HIDEPRINTTOFILE;  //打印配置界面的按鈕可用性,因為后台打印,其實這個配置沒什么意義  
  14.   
  15.     CPrintDialog pPrintdlg(FALSE, dwFlag, this);                                            //CPrintDialog實例化,因為MFC的打印設備無關性,可以理解為這就是一台打印機  
  16.   
  17.     HGLOBAL hDevMode = NULL;  
  18.     HGLOBAL hDevNames = NULL;  
  19.     if (GetPrinterDevice(strPrintDevice.GetBuffer(0), &hDevNames, &hDevMode))               //獲得指定打印機的配置、名字  
  20.         AfxGetApp()->SelectPrinter(hDevNames, hDevMode);  
  21.     else  
  22.         AfxMessageBox(_T("Failed to select custom printer"));  
  23.   
  24.     strPrintDevice.ReleaseBuffer();  
  25.   
  26.     pPrintdlg.m_pd.hDevMode = hDevMode;                                                     //讓pPrintdlg使用我們指定的打印機  
  27.     pPrintdlg.m_pd.hDevNames = hDevNames;  
  28.   
  29.     CDC dc;  
  30.     dc.Attach(pPrintdlg.CreatePrinterDC());                                                 //后台打印創建法,如果需要彈出打印對話框,請用DoModal  
  31.   
  32.     DOCINFO di;                                                                             //下面的內容網上很多,就不解釋了  
  33.     di.cbSize = sizeof(DOCINFO);  
  34.     di.lpszDocName = _T("有驅打印測試");  
  35.     di.lpszDatatype = NULL;  
  36.     di.lpszOutput = NULL;  
  37.     di.fwType = 0;  
  38.   
  39.     dc.StartDocW(&di);  
  40.     dc.StartPage();  
  41.     dc.SetMapMode(MM_TEXT);  
  42.   
  43.     CRect recPrint(0, 0, dc.GetDeviceCaps(LOGPIXELSX), dc.GetDeviceCaps(LOGPIXELSY));  
  44.     dc.DPtoLP(&recPrint);  
  45.     dc.SetWindowOrg(0, 0);  
  46.   
  47.     CFont newFont;  
  48.     VERIFY(newFont.CreatePointFont(120, _T("宋體"), &dc));  
  49.     CFont* oldFont = dc.SelectObject(&newFont);  
  50.   
  51.     dc.SetTextAlign(TA_TOP | TA_LEFT);  
  52.   
  53.     CString strPrint;  
  54.     int nIndex = 0;  
  55.     int x = 50;  
  56.     int y = 50;  
  57.     CSize textSize;  
  58.     textSize = dc.GetTextExtent(_T("00"), 2);                           //根據當前字體的寬、高,后面以此高度為行高  
  59.   
  60.     while ((nIndex = strMessage.Find(_T("\r\n"))) > -1)                  //將IDC_EDIT1編輯框中內容打印,支持換行,一次換行等於'\r\n',所以在開頭strMessage += _T("\r\n")  
  61.     {  
  62.         strPrint = strMessage.Left(nIndex);  
  63.         strMessage = strMessage.Mid(nIndex+2);  
  64.   
  65.         dc.TextOutW(x, y, strPrint);  
  66.   
  67.         y += textSize.cy;                                               //下移一行,行高為字體高度  
  68.     }  
  69.   
  70.     dc.SelectObject(oldFont);  
  71.     newFont.DeleteObject();  
  72.     dc.EndPage();  
  73.     dc.EndDoc();  
  74.     DeleteDC(dc.Detach());  


關於GetPrinterDevice,來自微軟的一篇文章,點我跳轉

 

代碼也貼出來

 

[cpp]  view plain copy 在CODE上查看代碼片 派生到我的代碼片
 
 
 
  1. BOOL CMFCApplication2Dlg::GetPrinterDevice(LPTSTR pszPrinterName, HGLOBAL* phDevNames, HGLOBAL* phDevMode)  
  2. {  
  3.     // if NULL is passed, then assume we are setting app object's  
  4.     // devmode and devnames  
  5.     if (phDevMode == NULL || phDevNames == NULL)  
  6.         return FALSE;  
  7.   
  8.     // Open printer  
  9.     HANDLE hPrinter;  
  10.     if (OpenPrinter(pszPrinterName, &hPrinter, NULL) == FALSE)  
  11.         return FALSE;  
  12.   
  13.     // obtain PRINTER_INFO_2 structure and close printer  
  14.     DWORD dwBytesReturned, dwBytesNeeded;  
  15.     GetPrinter(hPrinter, 2, NULL, 0, &dwBytesNeeded);  
  16.     PRINTER_INFO_2* p2 = (PRINTER_INFO_2*)GlobalAlloc(GPTR,  
  17.         dwBytesNeeded);  
  18.     if (GetPrinter(hPrinter, 2, (LPBYTE)p2, dwBytesNeeded,  
  19.         &dwBytesReturned) == 0) {  
  20.             GlobalFree(p2);  
  21.             ClosePrinter(hPrinter);  
  22.             return FALSE;  
  23.     }  
  24.     ClosePrinter(hPrinter);  
  25.   
  26.     // Allocate a global handle for DEVMODE  
  27.     HGLOBAL  hDevMode = GlobalAlloc(GHND, sizeof(*p2->pDevMode) +  
  28.         p2->pDevMode->dmDriverExtra);  
  29.     ASSERT(hDevMode);  
  30.     DEVMODE* pDevMode = (DEVMODE*)GlobalLock(hDevMode);  
  31.     ASSERT(pDevMode);  
  32.   
  33.     // copy DEVMODE data from PRINTER_INFO_2::pDevMode  
  34.     memcpy(pDevMode, p2->pDevMode, sizeof(*p2->pDevMode) +  
  35.         p2->pDevMode->dmDriverExtra);  
  36.     GlobalUnlock(hDevMode);  
  37.   
  38.     // Compute size of DEVNAMES structure from PRINTER_INFO_2's data  
  39.     DWORD drvNameLen = lstrlen(p2->pDriverName)+1;  // driver name  
  40.     DWORD ptrNameLen = lstrlen(p2->pPrinterName)+1; // printer name  
  41.     DWORD porNameLen = lstrlen(p2->pPortName)+1;    // port name  
  42.   
  43.     // Allocate a global handle big enough to hold DEVNAMES.  
  44.     HGLOBAL hDevNames = GlobalAlloc(GHND,  
  45.         sizeof(DEVNAMES) +  
  46.         (drvNameLen + ptrNameLen + porNameLen)*sizeof(TCHAR));  
  47.     ASSERT(hDevNames);  
  48.     DEVNAMES* pDevNames = (DEVNAMES*)GlobalLock(hDevNames);  
  49.     ASSERT(pDevNames);  
  50.   
  51.     // Copy the DEVNAMES information from PRINTER_INFO_2  
  52.     // tcOffset = TCHAR Offset into structure  
  53.     int tcOffset = sizeof(DEVNAMES)/sizeof(TCHAR);  
  54.     ASSERT(sizeof(DEVNAMES) == tcOffset*sizeof(TCHAR));  
  55.   
  56.     pDevNames->wDriverOffset = tcOffset;  
  57.     memcpy((LPTSTR)pDevNames + tcOffset, p2->pDriverName,  
  58.         drvNameLen*sizeof(TCHAR));  
  59.     tcOffset += drvNameLen;  
  60.   
  61.     pDevNames->wDeviceOffset = tcOffset;  
  62.     memcpy((LPTSTR)pDevNames + tcOffset, p2->pPrinterName,  
  63.         ptrNameLen*sizeof(TCHAR));  
  64.     tcOffset += ptrNameLen;  
  65.   
  66.     pDevNames->wOutputOffset = tcOffset;  
  67.     memcpy((LPTSTR)pDevNames + tcOffset, p2->pPortName,  
  68.         porNameLen*sizeof(TCHAR));  
  69.     pDevNames->wDefault = 0;  
  70.   
  71.     GlobalUnlock(hDevNames);  
  72.     GlobalFree(p2);   // free PRINTER_INFO_2  
  73.   
  74.     // set the new hDevMode and hDevNames  
  75.     *phDevMode = hDevMode;  
  76.     *phDevNames = hDevNames;  
  77.     return TRUE;  
  78. }  



 

基本上是完整代碼了,如果有內存錯誤,請聯系我


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM