技术演进中的开发沉思-38 MFC系列:关于打印
打印程序也是MFC开发中不能忽视的一个环节,现在做打印开发so easy。但当年做打印开发还是挺麻烦。在当年的桌面程序里就像拼图的最后一块,看着简单,实则要把屏幕上的像素世界,准确映射到打印机的物理纸张上。而MFC 的打印机制就像老照相馆的暗房:你不用懂显影液的配方(打印机驱动),但得知道怎么调整光圈(设备上下文),才能让照片(打印效果)和底片(屏幕显示)一致。
一、打印
MFC 用 “设备上下文”(DC)机制把这套流程封装得严丝合缝。CDC这个类就像个万能画板 —— 往pDC->m_hDC里塞屏幕设备句柄,它就是能显示彩色像素的电子画布;换成打印机句柄,就变成了能输出墨点的物理画布。我第一次掉坑里,是发现屏幕上清晰的折线图,打印出来竟缩成了左上角的小方块。对着调试器看了半天才明白:屏幕用的是 “像素坐标”,100 像素在 17 寸显示器上大概 1.5 厘米;而打印机认的是 “逻辑单位”,同样 100 单位在 A4 纸上能占 3 厘米。就像用惯了厘米尺的人突然换成英寸尺,比例没换算对,画出的东西自然走样。
解决这个问题时,我在OnPrepareDC里加了段坐标转换代码:
void CMyView::OnPrepareDC(CDC* pDC, CPrintInfo* pInfo){CView::OnPrepareDC(pDC, pInfo);if (pDC->IsPrinting()) {// 把打印机逻辑单位设为毫米pDC->SetMapMode(MM_LOMETRIC);// 调整原点到左上角(打印机默认原点在左下角)pDC->SetWindowOrg(0, -2970); // A4纸高度297毫米,转换为0.1毫米单位}}
这段代码让打印机突然 “看懂” 了屏幕坐标 —— 就像给两个说不同方言的人配了翻译。改完那天,李教授来试打印,看着纸张缓缓吐出时,他手指在图表边缘比了比:“这下对齐了,比我用尺子量着画强多了。”
二、MFC 的 “默认菜谱”
MFC 最贴心的地方,是把打印的基础流程做成了 “半成品菜”。就像超市里切好的净菜,你不用自己洗菜切菜(写设备初始化代码),只要按口味加点调料(重写绘图函数)就行。默认机制下,CView类的OnDraw函数是个多面手 —— 屏幕刷新时它被调用,打印时 MFC 会自动把打印机 DC 传进来,让同一段绘图代码在两种设备上生效。
早期做实验报告打印时,我就靠这个机制省了不少事。OnDraw里先画标题,再画数据表格,最后加实验员签名栏:
void CMyView::OnDraw(CDC* pDC){CMyDoc* pDoc = GetDocument();ASSERT_VALID(pDoc);if (!pDoc) return;// 标题用粗体CFont font, *pOldFont;font.CreatePointFont(160, _T("宋体"), pDC); // 16pt字体pOldFont = pDC->SelectObject(&font);pDC->TextOut(100, 50, _T("材料抗压实验报告"));pDC->SelectObject(pOldFont);// 画表格线CPen pen(PS_SOLID, 1, RGB(0,0,0)), *pOldPen;pOldPen = pDC->SelectObject(&pen);DrawTableBorder(pDC); // 自定义画边框函数pDC->SelectObject(pOldPen);// 填充数据DrawExperimentData(pDC, pDoc->m_dataList);}
但这套默认机制有个明显短板:它不知道 “一页该装多少内容”。有次李教授导入了三十组实验数据,结果打印机把所有表格挤在一页纸上,字小得要用放大镜看。这就像写文章忘了分段,用户看着费劲。后来我在OnPreparePrinting里加了分页逻辑:
BOOL CMyView::OnPreparePrinting(CPrintInfo* pInfo){if (!DoPreparePrinting(pInfo))return FALSE;// 计算总页数:每页最多显示8组数据CMyDoc* pDoc = GetDocument();int nTotalData = pDoc->m_dataList.GetCount();int nMaxPage = (nTotalData + 7) / 8; // 向上取整pInfo->SetMaxPage(nMaxPage > 0 ? nMaxPage : 1);return TRUE;}
加了这段代码后,数据会自动分到多页,每页底部还能加上页码。客户拿着打印好的报告翻了两页,突然说:“这比我当年用打字机方便多了 —— 那时候换页得手动卷纸。”
三、给 Scribble “加分页”
微软的 Scribble 示例程序,是我们那代程序员的 “技术启蒙教材”。这个能随手涂鸦的小程序,默认打印时却像把整本速写本强行压成一页纸 —— 如果用户画了条横贯三屏的曲线,打印出来就会变成细得像头发丝的线条。我们做项目时,给它加了三个 “增强包”,让它从 “便签纸” 变成能装订的 “画册”。
页设置对话框是第一个要加的。就像去打印店先选纸张,我用CPrintDialog做了个设置界面,让用户能选 A4 还是 B5,横版还是竖版。调试时发现个有趣的细节:当用户选 “横向”,MFC 会自动调整CPrintInfo里的纸张尺寸,这时候绘图坐标得跟着变。我在OnBeginPrinting里加了判断:
void CMyScribbleView::OnBeginPrinting(CDC* pDC, CPrintInfo* pInfo){CView::OnBeginPrinting(pDC, pInfo);// 记录纸张方向和尺寸m_bLandscape = (pInfo->m_pPD->m_pd.Flags & PD_LANDSCAPE) != 0;m_sizePaper = pInfo->m_sizePaper; // 单位是0.01毫米}
有次测试时,我选了横向打印,曲线突然跑到纸外面去了。查了半天才发现,他画的曲线坐标是按竖版纸张算的。后来我在绘图前加了坐标适配:如果是横向,就把 X 和 Y 轴的绘图范围对调,就像把画布旋转 90 度再下笔。
坐标映射是第二个重点。屏幕上用像素定位,打印机用物理单位,这中间得有个 “换算器”。SetMapMode函数就像把尺子刻度从 “像素” 换成 “毫米”,但实际用的时候要注意:打印机的原点默认在纸张左下角,而屏幕原点在左上角,不调整的话,画出来的东西会上下颠倒。我通常这么处理:
void CMyScribbleView::OnPrepareDC(CDC* pDC, CPrintInfo* pInfo){CView::OnPrepareDC(pDC, pInfo);if (pDC->IsPrinting()) {pDC->SetMapMode(MM_LOMETRIC); // 1单位=0.1毫米// 把原点移到左上角,Y轴向下为正pDC->SetWindowOrg(0, -m_sizePaper.cy);}}
这么一改,屏幕上从左到右画的线条,打印出来也是从左到右,不会变成 “镜像效果” 了。
智能分页是最费脑筋的。我给每个涂鸦线段加了 “高度标记”,就像在文章里标上段落长度。打印时程序会累计高度,超过纸张高度就自动分页。代码里用OnPrint代替OnDraw,根据当前页码决定画哪些内容:
void CMyScribbleView::OnPrint(CDC* pDC, CPrintInfo* pInfo){int nCurPage = pInfo->m_nCurPage;int nStartIndex = (nCurPage - 1) * m_nMaxLinesPerPage;int nEndIndex = min(nStartIndex + m_nMaxLinesPerPage, m_lines.GetSize());// 只绘制当前页包含的线段for (int i = nStartIndex; i < nEndIndex; i++) {DrawLine(pDC, m_lines[i]);}}
改完这个功能那天,我画了条从北京到上海的曲线(模拟铁路枢纽线路),程序自动分成 3 页打印,拼接起来刚好是完整的路线。这种 “技术实现想法” 的瞬间,大概就是程序员的快乐时刻。
四、打印预览
打印预览绝对是 MFC 里的 “温柔设计”。早年没这功能时,我们常为了调格式浪费半盒 A4 纸 —— 有时候差一毫米就偏出边框,得改一行代码重新编译,再打印出来看效果。预览功能就像寄信前先看一眼信封,在屏幕上就能看到最终效果。
MFC 的预览原理其实很巧妙:它先创建个内存 DC 当 “虚拟打印机”,把内容画到内存里,再缩放显示到屏幕窗口。CPrintPreviewState这个类会管理预览窗口的缩放比例,你点 “放大”,它就把内存里的图像按 1:1 显示;点 “缩小”,就按比例缩小,像用放大镜看地图。
我做报表系统时,加了个 “实时预览” 功能 —— 用户改了表格样式,预览窗口会立刻刷新。实现时用了OnUpdate消息:
void CMyReportView::OnUpdate(CView* pSender, LPARAM lHint, CObject* pHint){if (IsPreviewMode()) {// 如果在预览模式,强制刷新Invalidate();}CView::OnUpdate(pSender, lHint, pHint);}
这种方法实现后,客户调整表头格式,就只需要拖动鼠标调整列宽了,在没有打印功能的时候,很多图表,都是客户用尺子量着画表格的, 那时候改一次报表得重新抄一遍。而当年我们开发的应用改变了他们的工作方式,其实技术进步的意义,有时就是把人从机械劳动里解放出来。
最后小结:
现在想想,MFC 的打印机制,其实藏着一套产品思维:默认机制解决 80% 的基础需求(不用重复造轮子);增强功能应对个性化场景(给高手留发挥空间);预览功能则照顾到用户的 “安全感”(避免失误)。MFC 打印的可贵之处,在于它用类封装了复杂的设备交互逻辑,让我们能专注于业务需求 —— 这和现在的云打印 API 的设计理念不谋而合。当年调试OnPrint函数时领悟的 “设备无关性” 原则,现在依然适用于云打印开发:无论底层是激光还是喷墨,代码只需要关心 “打印什么”,而不是 “怎么打印”。技术会迭代,但解决问题的本质从未改变。就像从 MFC 到云打印,变的是实现手段,不变的是让用户 “用着顺手” 的追求。未完待续..........