c++ - 如何在Win32 C++中进行打印预览?

标签 c++ windows winapi gdi print-preview

我有一个只需使用HDC的绘图功能。
但是我需要显示将要打印的内容的精确缩放版本。

所以目前,我使用
具有打印机HDC的CreateCompatibleDC()和
带有打印机的HDC的CreateCompatibleBitmap()。

我以这种方式认为DC将具有打印机的确切宽度和高度。
当我在此HDC中选择字体时,文本将完全按照打印机的比例缩放。

不幸的是,我无法使用StretchBlt()将这个HDC的像素复制到控件的HDC,因为我猜它们是不同的HDC类型。

如果我从HDC窗口中以与打印机页面相同的宽度创建“内存 Canvas ”,
字体以小字体显示,因为它们是针对屏幕而不是页面缩放的。

我应该从窗口的DC创建CreateCompatibleDC()还是
从打印机的DC或其他的CreateCompatibleBitmap()?

如果有人可以解释执行此操作的正确方法。
(并且仍然具有与打印机上一样的外观)...

好吧,我将不胜感激!

...史蒂夫

最佳答案

根据您想要的精度,这可能会变得困难。

有很多方法。听起来您正在尝试绘制到打印机大小的位图,然后将其缩小。要执行的步骤是:

  • 为打印机创建一个DC(或者更好的是一个IC-Information Context)。
  • 查询打印机DC以找出分辨率,页面大小,物理偏移量等。
  • 为窗口/屏幕创建DC。
  • 创建兼容的DC(内存DC)。
  • 为窗口/屏幕创建兼容的位图,但大小应为打印机页面的像素大小。 (此方法的问题在于,这是一个巨大的位图,它可能会失败。)
  • 选择兼容的位图到内存DC中。
  • 使用与绘制到实际打印机时相同的坐标绘制到内存DC。 (选择字体时,请确保将其缩放到打印机的逻辑英寸,而不是屏幕的逻辑英寸。)
  • StretchBlt将内存DC编码到窗口,它将按比例缩小整个图像。您可能想尝试拉伸(stretch)模式,以了解哪种方式最适合您要显示的图像。
  • 释放所有资源。

  • 但是在朝这个方向前进之前,请考虑其他选择。此方法涉及分配HUGE屏幕外位图。这可能会在资源匮乏的计算机上失败。即使没有,您可能也会饿死其他应用程序。

    对于许多应用程序,另一个答案中给出的图元文件方法是一个不错的选择。我将从这个开始。

    另一种方法是找出某个虚构的高分辨率单位中的所有大小。例如,假设一切都在千分之一英寸内。然后,您的绘图例程会将这个虚拟单位缩放为目标设备所使用的实际dpi。

    最后一种方法(可能还有图元文件)的问题是GDI字体不能完美地线性缩放。根据目标分辨率调整各个字符的宽度。在高分辨率设备(例如300+ dpi激光打印机)上,这种调整是最小的。但是在96 dpi的屏幕上,这些调整可能会在一行的整个长度上造成很大的误差。因此,预览窗口中的文本可能会与打印页面上的文本不成比例(通常更宽)。

    因此,最核心的方法是在打印机环境中测量文本,然后在屏幕环境中再次测量,并针对差异进行调整。例如,(使用虚构数字),您可以测量打印机上下文中某些文本的宽度,得出的宽度为900个打印机像素。假设打印机像素与屏幕像素的比例为3:1。您希望屏幕上的同一文本为300屏幕像素宽。但是,您在屏幕上下文中进行测量,将获得325屏幕像素之类的值。绘制到屏幕上时,必须以某种方式使文本变窄25个像素。您可以将字符放在一起,或者选择稍小的字体,然后将其拉伸(stretch)。

    核心方法涉及更多的复杂性。例如,您可能尝试检测打印机驱动程序进行的字体替换,并使其与可用的屏幕字体尽可能地匹配。

    我很高兴将大位图和硬核方法结合在一起。我没有为整个页面制作一个巨大的位图,而是为一个文本行制作了一个足够大的位图。然后,我以打印机尺寸绘制到屏幕外的位图,并对其进行StretchBlt编码,直到达到屏幕尺寸。这样就消除了字体大小略有下降的大小差异。它适用于实际的打印预览,但是您不想构建这样的WYSIWYG编辑器。单行位图足够小以使其实用。

    好消息是只有文字很难。所有其他图形都是坐标和尺寸的简单缩放。

    我使用GDI +的次数不多,但我认为它消除了非线性字体缩放的问题。因此,如果您使用的是GDI +,则只需要缩放坐标即可。缺点是我认为GDI +上的字体质量不佳。

    最后,如果您是Vista或更高版本上的 native 应用程序,请确保已将进程标记为“DPI-aware”。否则,如果用户使用的是高DPI屏幕,则Windows会对您说谎,并声称分辨率仅为96 dpi,然后对所绘制的内容进行模糊放大。这会降低视觉质量,并使调试打印预览更加复杂。由于许多程序无法很好地适应更高的DPI屏幕,因此Microsoft默认在Vista中添加了“高DPI缩放比例”。

    编辑以添加

    另一个警告:如果您使用打印机大小的位图将HFONT选择到内存DC中,则可能会获得与将相同的HFONT选择到实际打印机DC中时不同的字体。这是因为某些打印机驱动程序将用内存字体替换常见字体。例如,某些PostScript打印机将内部PostScript字体替换为某些常见的TrueType字体。

    您可以先在打印机IC中选择HFONT,然后使用GetTextFaceGetTextMetrics以及GetOutlineTextMetrics等GDI函数来查找有关所选实际字体的信息。然后,您可以创建一个新的LOGFONT,以尝试更紧密地匹配打印机使用的内容,将其转换为HFONT,然后将其选择到内存DC中。这是一个非常好的实现的标志。

    关于c++ - 如何在Win32 C++中进行打印预览?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2240243/

    相关文章:

    python - 通过套接字传输大文件

    C++(没有 MFC)禁用/灰色菜单项

    c++ - 将数据从 NodeJS 传输到 C++ 应用程序

    c++ - 将 unique_ptr 插入 map ,指针被销毁

    在 C 中将 UNICODE_STRING 转换为 ANSI,反之亦然

    windows - awssync s3 Windows 计划任务返回 0x1

    c++ - 实现前缀运算符++

    c++ - 如何格式化输出或显示?

    windows - Inno Setup - 在 "Send to"上下文菜单中创建快捷方式

    c++ - 在 OO 中创建一个窗口