推荐下 不错。

对话框打印,网上一搜一大堆,基本分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. EnumPrinters(dwFlag, NULL, 4, NULL, 0, &dwNeeded, &dwReturn);
  6. PRINTER_INFO_4* p4;
  7. p4 = new PRINTER_INFO_4[dwNeeded];
  8. EnumPrinters(dwFlag, NULL, 4, (PBYTE)p4, dwNeeded, &dwNeeded, &dwReturn);
  9. for (int i = 0; i<(int)dwReturn; i++)
  10. this->m_cboPrint.AddString(p4[i].pPrinterName);
  11. delete []p4;

3、操作对话框,在IDC_EDIT1里输入打印的内容,在IDC_COMBO1里选中打印机;

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

  1. // TODO: 在此添加控件通知处理程序代码
  2. //  CDialogEx::OnOK();
  3. this->UpdateData();
  4. CString strMessage;
  5. CString strPrintDevice;
  6. this->GetDlgItem(IDC_EDIT1)->GetWindowTextW(strMessage);
  7. strMessage += _T("\r\n");       //添加结尾,方便后面循环读取打印数据
  8. this->GetDlgItem(IDC_COMBO1)->GetWindowTextW(strPrintDevice);
  9. DWORD dwFlag = PD_ALLPAGES | PD_NOPAGENUMS | PD_USEDEVMODECOPIES | PD_HIDEPRINTTOFILE;  //打印配置界面的按钮可用性,因为后台打印,其实这个配置没什么意义
  10. CPrintDialog pPrintdlg(FALSE, dwFlag, this);                                            //CPrintDialog实例化,因为MFC的打印设备无关性,可以理解为这就是一台打印机
  11. HGLOBAL hDevMode = NULL;
  12. HGLOBAL hDevNames = NULL;
  13. if (GetPrinterDevice(strPrintDevice.GetBuffer(0), &hDevNames, &hDevMode))               //获得指定打印机的配置、名字
  14. AfxGetApp()->SelectPrinter(hDevNames, hDevMode);
  15. else
  16. AfxMessageBox(_T("Failed to select custom printer"));
  17. strPrintDevice.ReleaseBuffer();
  18. pPrintdlg.m_pd.hDevMode = hDevMode;                                                     //让pPrintdlg使用我们指定的打印机
  19. pPrintdlg.m_pd.hDevNames = hDevNames;
  20. CDC dc;
  21. dc.Attach(pPrintdlg.CreatePrinterDC());                                                 //后台打印创建法,如果需要弹出打印对话框,请用DoModal
  22. DOCINFO di;                                                                             //下面的内容网上很多,就不解释了
  23. di.cbSize = sizeof(DOCINFO);
  24. di.lpszDocName = _T("有驱打印测试");
  25. di.lpszDatatype = NULL;
  26. di.lpszOutput = NULL;
  27. di.fwType = 0;
  28. dc.StartDocW(&di);
  29. dc.StartPage();
  30. dc.SetMapMode(MM_TEXT);
  31. CRect recPrint(0, 0, dc.GetDeviceCaps(LOGPIXELSX), dc.GetDeviceCaps(LOGPIXELSY));
  32. dc.DPtoLP(&recPrint);
  33. dc.SetWindowOrg(0, 0);
  34. CFont newFont;
  35. VERIFY(newFont.CreatePointFont(120, _T("宋体"), &dc));
  36. CFont* oldFont = dc.SelectObject(&newFont);
  37. dc.SetTextAlign(TA_TOP | TA_LEFT);
  38. CString strPrint;
  39. int nIndex = 0;
  40. int x = 50;
  41. int y = 50;
  42. CSize textSize;
  43. textSize = dc.GetTextExtent(_T("00"), 2);                           //根据当前字体的宽、高,后面以此高度为行高
  44. while ((nIndex = strMessage.Find(_T("\r\n"))) > -1)                  //将IDC_EDIT1编辑框中内容打印,支持换行,一次换行等于'\r\n',所以在开头strMessage += _T("\r\n")
  45. {
  46. strPrint = strMessage.Left(nIndex);
  47. strMessage = strMessage.Mid(nIndex+2);
  48. dc.TextOutW(x, y, strPrint);
  49. y += textSize.cy;                                               //下移一行,行高为字体高度
  50. }
  51. dc.SelectObject(oldFont);
  52. newFont.DeleteObject();
  53. dc.EndPage();
  54. dc.EndDoc();
  55. DeleteDC(dc.Detach());

关于GetPrinterDevice,来自微软的一篇文章,点我跳转

代码也贴出来

  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. // Open printer
  8. HANDLE hPrinter;
  9. if (OpenPrinter(pszPrinterName, &hPrinter, NULL) == FALSE)
  10. return FALSE;
  11. // obtain PRINTER_INFO_2 structure and close printer
  12. DWORD dwBytesReturned, dwBytesNeeded;
  13. GetPrinter(hPrinter, 2, NULL, 0, &dwBytesNeeded);
  14. PRINTER_INFO_2* p2 = (PRINTER_INFO_2*)GlobalAlloc(GPTR,
  15. dwBytesNeeded);
  16. if (GetPrinter(hPrinter, 2, (LPBYTE)p2, dwBytesNeeded,
  17. &dwBytesReturned) == 0) {
  18. GlobalFree(p2);
  19. ClosePrinter(hPrinter);
  20. return FALSE;
  21. }
  22. ClosePrinter(hPrinter);
  23. // Allocate a global handle for DEVMODE
  24. HGLOBAL  hDevMode = GlobalAlloc(GHND, sizeof(*p2->pDevMode) +
  25. p2->pDevMode->dmDriverExtra);
  26. ASSERT(hDevMode);
  27. DEVMODE* pDevMode = (DEVMODE*)GlobalLock(hDevMode);
  28. ASSERT(pDevMode);
  29. // copy DEVMODE data from PRINTER_INFO_2::pDevMode
  30. memcpy(pDevMode, p2->pDevMode, sizeof(*p2->pDevMode) +
  31. p2->pDevMode->dmDriverExtra);
  32. GlobalUnlock(hDevMode);
  33. // Compute size of DEVNAMES structure from PRINTER_INFO_2's data
  34. DWORD drvNameLen = lstrlen(p2->pDriverName)+1;  // driver name
  35. DWORD ptrNameLen = lstrlen(p2->pPrinterName)+1; // printer name
  36. DWORD porNameLen = lstrlen(p2->pPortName)+1;    // port name
  37. // Allocate a global handle big enough to hold DEVNAMES.
  38. HGLOBAL hDevNames = GlobalAlloc(GHND,
  39. sizeof(DEVNAMES) +
  40. (drvNameLen + ptrNameLen + porNameLen)*sizeof(TCHAR));
  41. ASSERT(hDevNames);
  42. DEVNAMES* pDevNames = (DEVNAMES*)GlobalLock(hDevNames);
  43. ASSERT(pDevNames);
  44. // Copy the DEVNAMES information from PRINTER_INFO_2
  45. // tcOffset = TCHAR Offset into structure
  46. int tcOffset = sizeof(DEVNAMES)/sizeof(TCHAR);
  47. ASSERT(sizeof(DEVNAMES) == tcOffset*sizeof(TCHAR));
  48. pDevNames->wDriverOffset = tcOffset;
  49. memcpy((LPTSTR)pDevNames + tcOffset, p2->pDriverName,
  50. drvNameLen*sizeof(TCHAR));
  51. tcOffset += drvNameLen;
  52. pDevNames->wDeviceOffset = tcOffset;
  53. memcpy((LPTSTR)pDevNames + tcOffset, p2->pPrinterName,
  54. ptrNameLen*sizeof(TCHAR));
  55. tcOffset += ptrNameLen;
  56. pDevNames->wOutputOffset = tcOffset;
  57. memcpy((LPTSTR)pDevNames + tcOffset, p2->pPortName,
  58. porNameLen*sizeof(TCHAR));
  59. pDevNames->wDefault = 0;
  60. GlobalUnlock(hDevNames);
  61. GlobalFree(p2);   // free PRINTER_INFO_2
  62. // set the new hDevMode and hDevNames
  63. *phDevMode = hDevMode;
  64. *phDevNames = hDevNames;
  65. return TRUE;
  66. }

基本上是完整代码了,如果有内存错误,请联系我

MFC对话框使用CPrintDialog实现打印,指定打印机、后台打印的更多相关文章

  1. js 打印指定页面部分打印

    <!DOCTYPE html><html>    <head>        <meta charset="UTF-8">      ...

  2. C#通过调用WinApi打印PDF文档类,服务器PDF打印、IIS PDF打印

    其他网站下载来的类,可以用于Winform.Asp.Net,用于服务器端PDF或其他文件打印. 直接上代码: using System; using System.Collections.Generi ...

  3. 根据第三方库spire.pdf使用指定打印机打印pdf文件

    private void button1_Click(object sender, EventArgs e) { PdfDocument doc = new PdfDocument(); string ...

  4. 打印机设置(PrintDialog)、页面设置(PageSetupDialog) 及 RDLC报表如何选择指定打印机

    如果一台电脑同时连接多个打印机,而且每个打印机使用的纸张大小各不相同(比如:票据打印钱用的小票专用张,办公打印机用的是A4标准纸),在处理打印类的需求时,如果不用代码干预,用户必须每次打印时,都必须在 ...

  5. Java调用TSC打印机进行打印

    最近项目中用到了打印机,最开始的完全不懂,现在弄好了,所以做了总结,该篇包括后台的调用打印(两种方式)跟前端的js的打印,但是只有IE现在支持打印,而且如果想远程连接打印机,二维码的生成和直接由打印机 ...

  6. MFC对话框显示BMP图片

    1.MFC对话框显示BMP图片我们先从简单的开始吧.先分一个类: (一) 非动态显示图片(即图片先通过资源管理器载入,有一个固定ID) (二) 动态载入图片(即只需要在程序中指定图片的路径即可载入) ...

  7. MFC对话框中显示BMP,JPG图片

    //************************************ // 方法说明:    显示JPG和GIF.BMP图片 // 参数说明:    CDC * pDC           设 ...

  8. MFC对话框

    创建对话框步骤: 第一,创建对话框资源,主要包括创建新的对话框模板.设置对话框属性和为对话框添加各种控件: 第二,生成对话框类,主要包括新建对话框类.添加控件变量和控件的消息处理函数等. 创建对话框类 ...

  9. 使用Windows驱动的虚拟打印机,打印Excel表格无表格线问题解决(2)

    测试: 经前天的测试,最终还是没有明显的定夺到底是驱动的问题,还是打印机的问题.但是按照可能性来排查,最明显的一点就是其他测试环境不变的情况下增加一张图片,就可以打印出表格线,我始终觉得这里是突破点, ...

随机推荐

  1. vue中使用element写点击input内部标签(使用模态框传值)

    首先附上源码地址 https://files.cnblogs.com/files/maruihua/vue-tagsinput-master.zip 这个是我修改后的代码.取消了部分功能,添加的一些功 ...

  2. POJ2482 Stars in Your Window(扫描线+区间最大+区间更新)

    Fleeting time does not blur my memory of you. Can it really be 4 years since I first saw you? I stil ...

  3. 使用VS2008,VS2010编译64位的应用程序

    要编译生成64位的应用程序,就必须把vs2008,或vs2010的配置管理器设置为x64. 如果你的配置管理器那里没有x64这个选项,那么是你在安装vs时可能没有安装这个组件.你不用卸载vs,只需打开 ...

  4. Qt对象模型之一:信号和槽

    一.信号和槽机制概述 信号槽是 Qt 框架引以为豪的机制之一.所谓信号槽,实际就是观察者模式.当某个事件发生之后,比如,按钮检测到自己被点击了一下,它就会发出一个信号(signal).这种发出是没有目 ...

  5. C#课程设计

    一.问题描述 监视文件和文件夹的变化(FileSystemWatcher 类):选择一个文件夹,单击"开始监视",该文件夹处于监视状态,此时对该文件夹如果进行操作,这些行为将显示出 ...

  6. (转)C语言运算符优先级 详细列表

    C语言运算符优先级 详细列表 文章转自:Slyar Home 优先级 运算符 名称或含义 使用形式 结合方向 说明 1 [] 数组下标 数组名[常量表达式] 左到右   () 圆括号 (表达式)/函数 ...

  7. jmeter(十五)Jmeter默认报告优化

    一.本文目的: 之前写了两篇文章搭建持续集成接口测试平台(Jenkins+Ant+Jmeter)和ANT批量执行Jmeter脚本,功能实现上都没有什么问题,但是最后生成的报告有一点小问题,虽然不影响使 ...

  8. B. Hongcow Solves A Puzzle

    http://codeforces.com/contest/745/problem/B 题目要求的是,给定一个图形,要求里面判断是否有矩形,且仅有一个 就是 XXX.... XXX...X 是不行的, ...

  9. android开发学习——facebook第三方登录,看了你不会后悔

    给APP用原生android进行facebook第三方登录. 我们做一件事情,首先得了解其原理,这样才不会迷茫,才知道自己做到什么程度了,心里才会有底. 所以,第一步,了解第三方登录的原理:下面贴一些 ...

  10. Java_JDBC连接数据库_使用读取配置文件的方式

    package com.homewoek3_4.dao; import java.io.IOException; import java.io.InputStream; import java.sql ...