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函数,第一次的到结构体的大小,第二次得到打印机列表
- // TODO: 在此添加额外的初始化代码
- DWORD dwNeeded;
- DWORD dwReturn;
- DWORD dwFlag = PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL;
- EnumPrinters(dwFlag, NULL, 4, NULL, 0, &dwNeeded, &dwReturn);
- PRINTER_INFO_4* p4;
- p4 = new PRINTER_INFO_4[dwNeeded];
- EnumPrinters(dwFlag, NULL, 4, (PBYTE)p4, dwNeeded, &dwNeeded, &dwReturn);
- for (int i = 0; i<(int)dwReturn; i++)
- this->m_cboPrint.AddString(p4[i].pPrinterName);
- delete []p4;
3、操作对话框,在IDC_EDIT1里输入打印的内容,在IDC_COMBO1里选中打印机;
4、打印(我是用OK按钮打印的,大家随便)
- // TODO: 在此添加控件通知处理程序代码
- // CDialogEx::OnOK();
- this->UpdateData();
- CString strMessage;
- CString strPrintDevice;
- this->GetDlgItem(IDC_EDIT1)->GetWindowTextW(strMessage);
- strMessage += _T("\r\n"); //添加结尾,方便后面循环读取打印数据
- this->GetDlgItem(IDC_COMBO1)->GetWindowTextW(strPrintDevice);
- DWORD dwFlag = PD_ALLPAGES | PD_NOPAGENUMS | PD_USEDEVMODECOPIES | PD_HIDEPRINTTOFILE; //打印配置界面的按钮可用性,因为后台打印,其实这个配置没什么意义
- CPrintDialog pPrintdlg(FALSE, dwFlag, this); //CPrintDialog实例化,因为MFC的打印设备无关性,可以理解为这就是一台打印机
- HGLOBAL hDevMode = NULL;
- HGLOBAL hDevNames = NULL;
- if (GetPrinterDevice(strPrintDevice.GetBuffer(0), &hDevNames, &hDevMode)) //获得指定打印机的配置、名字
- AfxGetApp()->SelectPrinter(hDevNames, hDevMode);
- else
- AfxMessageBox(_T("Failed to select custom printer"));
- strPrintDevice.ReleaseBuffer();
- pPrintdlg.m_pd.hDevMode = hDevMode; //让pPrintdlg使用我们指定的打印机
- pPrintdlg.m_pd.hDevNames = hDevNames;
- CDC dc;
- dc.Attach(pPrintdlg.CreatePrinterDC()); //后台打印创建法,如果需要弹出打印对话框,请用DoModal
- DOCINFO di; //下面的内容网上很多,就不解释了
- di.cbSize = sizeof(DOCINFO);
- di.lpszDocName = _T("有驱打印测试");
- di.lpszDatatype = NULL;
- di.lpszOutput = NULL;
- di.fwType = 0;
- dc.StartDocW(&di);
- dc.StartPage();
- dc.SetMapMode(MM_TEXT);
- CRect recPrint(0, 0, dc.GetDeviceCaps(LOGPIXELSX), dc.GetDeviceCaps(LOGPIXELSY));
- dc.DPtoLP(&recPrint);
- dc.SetWindowOrg(0, 0);
- CFont newFont;
- VERIFY(newFont.CreatePointFont(120, _T("宋体"), &dc));
- CFont* oldFont = dc.SelectObject(&newFont);
- dc.SetTextAlign(TA_TOP | TA_LEFT);
- CString strPrint;
- int nIndex = 0;
- int x = 50;
- int y = 50;
- CSize textSize;
- textSize = dc.GetTextExtent(_T("00"), 2); //根据当前字体的宽、高,后面以此高度为行高
- while ((nIndex = strMessage.Find(_T("\r\n"))) > -1) //将IDC_EDIT1编辑框中内容打印,支持换行,一次换行等于'\r\n',所以在开头strMessage += _T("\r\n")
- {
- strPrint = strMessage.Left(nIndex);
- strMessage = strMessage.Mid(nIndex+2);
- dc.TextOutW(x, y, strPrint);
- y += textSize.cy; //下移一行,行高为字体高度
- }
- dc.SelectObject(oldFont);
- newFont.DeleteObject();
- dc.EndPage();
- dc.EndDoc();
- DeleteDC(dc.Detach());
关于GetPrinterDevice,来自微软的一篇文章,点我跳转
代码也贴出来
- BOOL CMFCApplication2Dlg::GetPrinterDevice(LPTSTR pszPrinterName, HGLOBAL* phDevNames, HGLOBAL* phDevMode)
- {
- // if NULL is passed, then assume we are setting app object's
- // devmode and devnames
- if (phDevMode == NULL || phDevNames == NULL)
- return FALSE;
- // Open printer
- HANDLE hPrinter;
- if (OpenPrinter(pszPrinterName, &hPrinter, NULL) == FALSE)
- return FALSE;
- // obtain PRINTER_INFO_2 structure and close printer
- DWORD dwBytesReturned, dwBytesNeeded;
- GetPrinter(hPrinter, 2, NULL, 0, &dwBytesNeeded);
- PRINTER_INFO_2* p2 = (PRINTER_INFO_2*)GlobalAlloc(GPTR,
- dwBytesNeeded);
- if (GetPrinter(hPrinter, 2, (LPBYTE)p2, dwBytesNeeded,
- &dwBytesReturned) == 0) {
- GlobalFree(p2);
- ClosePrinter(hPrinter);
- return FALSE;
- }
- ClosePrinter(hPrinter);
- // Allocate a global handle for DEVMODE
- HGLOBAL hDevMode = GlobalAlloc(GHND, sizeof(*p2->pDevMode) +
- p2->pDevMode->dmDriverExtra);
- ASSERT(hDevMode);
- DEVMODE* pDevMode = (DEVMODE*)GlobalLock(hDevMode);
- ASSERT(pDevMode);
- // copy DEVMODE data from PRINTER_INFO_2::pDevMode
- memcpy(pDevMode, p2->pDevMode, sizeof(*p2->pDevMode) +
- p2->pDevMode->dmDriverExtra);
- GlobalUnlock(hDevMode);
- // Compute size of DEVNAMES structure from PRINTER_INFO_2's data
- DWORD drvNameLen = lstrlen(p2->pDriverName)+1; // driver name
- DWORD ptrNameLen = lstrlen(p2->pPrinterName)+1; // printer name
- DWORD porNameLen = lstrlen(p2->pPortName)+1; // port name
- // Allocate a global handle big enough to hold DEVNAMES.
- HGLOBAL hDevNames = GlobalAlloc(GHND,
- sizeof(DEVNAMES) +
- (drvNameLen + ptrNameLen + porNameLen)*sizeof(TCHAR));
- ASSERT(hDevNames);
- DEVNAMES* pDevNames = (DEVNAMES*)GlobalLock(hDevNames);
- ASSERT(pDevNames);
- // Copy the DEVNAMES information from PRINTER_INFO_2
- // tcOffset = TCHAR Offset into structure
- int tcOffset = sizeof(DEVNAMES)/sizeof(TCHAR);
- ASSERT(sizeof(DEVNAMES) == tcOffset*sizeof(TCHAR));
- pDevNames->wDriverOffset = tcOffset;
- memcpy((LPTSTR)pDevNames + tcOffset, p2->pDriverName,
- drvNameLen*sizeof(TCHAR));
- tcOffset += drvNameLen;
- pDevNames->wDeviceOffset = tcOffset;
- memcpy((LPTSTR)pDevNames + tcOffset, p2->pPrinterName,
- ptrNameLen*sizeof(TCHAR));
- tcOffset += ptrNameLen;
- pDevNames->wOutputOffset = tcOffset;
- memcpy((LPTSTR)pDevNames + tcOffset, p2->pPortName,
- porNameLen*sizeof(TCHAR));
- pDevNames->wDefault = 0;
- GlobalUnlock(hDevNames);
- GlobalFree(p2); // free PRINTER_INFO_2
- // set the new hDevMode and hDevNames
- *phDevMode = hDevMode;
- *phDevNames = hDevNames;
- return TRUE;
- }
基本上是完整代码了,如果有内存错误,请联系我
MFC对话框使用CPrintDialog实现打印,指定打印机、后台打印的更多相关文章
- js 打印指定页面部分打印
<!DOCTYPE html><html> <head> <meta charset="UTF-8"> ...
- C#通过调用WinApi打印PDF文档类,服务器PDF打印、IIS PDF打印
其他网站下载来的类,可以用于Winform.Asp.Net,用于服务器端PDF或其他文件打印. 直接上代码: using System; using System.Collections.Generi ...
- 根据第三方库spire.pdf使用指定打印机打印pdf文件
private void button1_Click(object sender, EventArgs e) { PdfDocument doc = new PdfDocument(); string ...
- 打印机设置(PrintDialog)、页面设置(PageSetupDialog) 及 RDLC报表如何选择指定打印机
如果一台电脑同时连接多个打印机,而且每个打印机使用的纸张大小各不相同(比如:票据打印钱用的小票专用张,办公打印机用的是A4标准纸),在处理打印类的需求时,如果不用代码干预,用户必须每次打印时,都必须在 ...
- Java调用TSC打印机进行打印
最近项目中用到了打印机,最开始的完全不懂,现在弄好了,所以做了总结,该篇包括后台的调用打印(两种方式)跟前端的js的打印,但是只有IE现在支持打印,而且如果想远程连接打印机,二维码的生成和直接由打印机 ...
- MFC对话框显示BMP图片
1.MFC对话框显示BMP图片我们先从简单的开始吧.先分一个类: (一) 非动态显示图片(即图片先通过资源管理器载入,有一个固定ID) (二) 动态载入图片(即只需要在程序中指定图片的路径即可载入) ...
- MFC对话框中显示BMP,JPG图片
//************************************ // 方法说明: 显示JPG和GIF.BMP图片 // 参数说明: CDC * pDC 设 ...
- MFC对话框
创建对话框步骤: 第一,创建对话框资源,主要包括创建新的对话框模板.设置对话框属性和为对话框添加各种控件: 第二,生成对话框类,主要包括新建对话框类.添加控件变量和控件的消息处理函数等. 创建对话框类 ...
- 使用Windows驱动的虚拟打印机,打印Excel表格无表格线问题解决(2)
测试: 经前天的测试,最终还是没有明显的定夺到底是驱动的问题,还是打印机的问题.但是按照可能性来排查,最明显的一点就是其他测试环境不变的情况下增加一张图片,就可以打印出表格线,我始终觉得这里是突破点, ...
随机推荐
- SQL 初级教程学习(二)
1.SQL 语句从 "Websites" 表中选取头两条记录: SELECT * FROM Websites LIMIT 2; SELECT TOP 50 PERCENT * FR ...
- 数据结构 - 顺序队列的实行(C语言)
数据结构-顺序队列的实现 1 顺序队列的定义 线性表有顺序存储和链式存储,队列作为一种特殊的线性表,也同样存在这两种存储方式.我们先来看队列的顺序存储结构. 队列的顺序储存结构:用数组存储队列,为了避 ...
- [poj3744] Scout YYF I【概率dp 数学期望】
传送门:http://poj.org/problem?id=3744 令f(i)表示到i,安全的概率.则f(i) = f(i - 1) * p + f(i - 2) * (1 - p),若i位置有地雷 ...
- uwp选取文件夹并读取其中的图片
uwp对文件的操作和wpf,winform等等有很大的不同,主要原因是uwp对权限的要求比较严格,不能想从前那样随心所欲的读取文件. 1.首先找到Package.appxmanifest这个文件,在功 ...
- [C#基础知识系列]专题十:全面解析可空类型[转]
原文链接 主要内容: 1:空合并操作符(?? 操作符) ??操作符也就是"空合并操作符",它代表的意思是两个操作数,如果左边的数不为null时,就返回左边的数,如果左边的数为nul ...
- T4870 水灾(sliker.cpp/c/pas) 1000MS 64MB
题目描述 大雨应经下了几天雨,却还是没有停的样子.土豪CCY刚从外地赚完1e元回来,知道不久除了自己别墅,其他的地方都将会被洪水淹没. CCY所在的城市可以用一个N*M(N,M<=50)的地图表 ...
- iOS Programming UINavigationController
iOS Programming UINavigationController the Settings application has multiple related screens of info ...
- C#枚举中的位运算权限分配
什么是位运算 常用的位运算主要有与(&), 或(|)和非(~), 比如: & = ; | = ; ~ = ; 运用在权限设计中 先建立一个枚举表示所有的权限管理操作: [Flags] ...
- vba,设置,excel,wps ,页面设置例子
Sub yemian()'按钮1点击触发 执行下面的命令With Sheets(1).PageSetup'对像是表格1的页面设置的函数PageSetup,下面是函数的属性修改前面加. .Orienta ...
- 面向对象编程(OOP)基础知识(一)
Java是一个支持并发.基于类和面向对象的计算机编程语言. 下面列出了面向对象软件开发的优点: 1.代码开发模块化,更易维护和修改. 2.代码复用. 3.增强代码的可靠性和灵活性. 4.增加代码的可理 ...