c++ - 如何获取打印机的DVTARGETDEVICE以使用ITextHost实现打印预览

标签 c++ winapi printing mfc ole

除了绘制文本之外,我仍然在调整 CRect 的大小(是的,实际上使用 MFC)方面遇到问题。

大小计算和绘制实现都需要 DVTARGETDEVICE。这是我的代码:

void CRichDrawText::DrawText(CDC& dc, const CRect& rect)
{
  // Draw the text in the windowless control onto the given device context,
  // within the given bounding rectangle.
  //HTRACE("%s w(%d) h(%d)\n", __FUNCTION__, rect.right - rect.left, rect.bottom - rect.top);
  RECTL rc = { rect.left, rect.top, rect.right, rect.bottom };
  LONG lViewID = 0;
  SIZE sPrev;
  bool bUnscale = false;
  if (dc.IsPrinting())
  {
      //lViewID = TXTVIEW_INACTIVE;
      if ((::GetDeviceCaps(dc.m_hDC, TECHNOLOGY) != DT_METAFILE) && (dc.m_hAttribDC != dc.m_hDC))
      {
          VERIFY(::ScaleWindowExtEx(dc.m_hDC,
                                                ::GetDeviceCaps(dc.m_hDC, LOGPIXELSX),
                                                ::GetDeviceCaps(dc.m_hAttribDC, LOGPIXELSX),
                                                ::GetDeviceCaps(dc.m_hDC, LOGPIXELSY),
                                                ::GetDeviceCaps(dc.m_hAttribDC, LOGPIXELSY), &sPrev));
          bUnscale = true;
     }
  }
  HRESULT hr = m_TextServ->TxDraw(DVASPECT_CONTENT,0,NULL,NULL,dc.GetSafeHdc(), bUnscale ? dc.m_hAttribDC : NULL,&rc,NULL,NULL,NULL,0, lViewID);
  if (bUnscale)
  {
      ::SetWindowExtEx(dc.m_hDC, sPrev.cx, sPrev.cy, NULL);
  }
  ASSERT(SUCCEEDED(hr));
}

它在屏幕上呈现出很棒的效果: Screen Rendering

但在打印到打印机或打印预览时不会。但它断言打印预览。 hr 等于 E_INVALIDARG,我猜测是因为我没有提供 DVTARGETDEVICE 参数。

如何获取 DVTARGETDEVICE?

我在调用堆栈的更上方有 MFC CPrintInfo,我可以将其传递给 DrawText 方法,这意味着我有 PRINTDLG 结构。我不是一个 OLE 人员,只是想在我的 CView 派生类中呈现富文本(多个 View 、分割 View ,全部处于不同的缩放级别!)

编辑 - 更多信息

在对 TxDraw 的调用中,如果我为 hicTargetDev 参数传入 NULL,我将不再获得 ASSERT,但随后我的打印预览会根据我的屏幕而不是打印机进行绘制 - 没有缩放,什么也没有: enter image description here

当我调整“打印预览”窗口的大小时,RTF 文本不会缩放,但其他所有内容都会正确缩放。

我知道我需要提供第二个 DC 句柄,以及有关打印机的信息(即 DVTARGETDEVICE)

更多信息: 嗯,我在代码项目上找到了这个。有一个隐藏在 MFC 源代码中的方法在\Microsoft Visual Studio 9.0\VC\atlmfc\src\mfc\olemisc.cpp

中未公开提供

DVTARGETDEVICE* AFXAPI _AfxOleCreateTargetDevice(LPPRINTDLG lpPrintDlg)

但是,打印预览仍然没有缩放。它的行为是一样的。这是我当前的实现:

void CRichDrawText::DrawText(CDC& dc, const CRect& rect, PRINTDLG *pPD, bool bPrintPreview)
{
    // Draw the text in the windowless control onto the given device context,
    // within the given bounding rectangle.
    //HTRACE("%s w(%d) h(%d)\n", __FUNCTION__, rect.right - rect.left, rect.bottom - rect.top);
    RECTL rc = { rect.left, rect.top, rect.right, rect.bottom };
    TXTVIEW eTV = TXTVIEW_ACTIVE;
    SIZE sPrev;
    bool bUnscale = false;

    // print functionality
    DVTARGETDEVICE *pTargetDevice = NULL;
    if (dc.IsPrinting())
    {
        HASSERT(pPD);
        if (pPD && (::GetDeviceCaps(dc.m_hDC, TECHNOLOGY) != DT_METAFILE) && (dc.m_hAttribDC != dc.m_hDC))
        {
            pTargetDevice = ::HEI_AfxOleCreateTargetDevice(pPD);
            if (pTargetDevice)
            {
                VERIFY(::ScaleWindowExtEx(dc.m_hDC,
                    ::GetDeviceCaps(dc.m_hDC, LOGPIXELSX),
                    ::GetDeviceCaps(dc.m_hAttribDC, LOGPIXELSX),
                    ::GetDeviceCaps(dc.m_hDC, LOGPIXELSY),
                    ::GetDeviceCaps(dc.m_hAttribDC, LOGPIXELSY), &sPrev));
                bUnscale = true;
                if (bPrintPreview)
                {
                    eTV = TXTVIEW_INACTIVE;
                }
            }
        }
        HASSERT(bUnscale);
    }

    HRESULT hr = m_TextServ->TxDraw(bUnscale ? DVASPECT_DOCPRINT : DVASPECT_CONTENT, 0, NULL, pTargetDevice, dc.GetSafeHdc(), bUnscale ? dc.m_hAttribDC : NULL, &rc, NULL, NULL, NULL, 0, eTV);
    ASSERT(SUCCEEDED(hr));
    if (bUnscale)
    {
        ::SetWindowExtEx(dc.m_hDC, sPrev.cx, sPrev.cy, NULL);
    }
    if (pTargetDevice != NULL)
        CoTaskMemFree(pTargetDevice);
}

从技术上讲,这可以编译并运行,所以自从我回答了这个问题后,我应该创建一个新问题。

最佳答案

我弄清楚了如何通过 _AfxOleCreateTargetDevice 创建 DVTARGETDEVICE(请参阅更多信息:上面的问题编辑),但打印预览仍然不起作用。

关于c++ - 如何获取打印机的DVTARGETDEVICE以使用ITextHost实现打印预览,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59365237/

相关文章:

c++ - CUDA 语法错误 '<'

c++ - 可以函数返回 0 作为引用

windows - OpenProcessToken 在 Windows XP 上因访问被拒绝 (5) 而失败

c - 向内存映射文件添加新字节

python - 在 Python 的一行中随机选择一个单词?

c++ - C++ 当前日期和时间的自定义格式

c++ - 条件语句的按位与异或

python - 如何使用 Python 将按钮(加载项)添加到 Outlook

c++ - 从文件中的特定位置读取 C++

有人可以帮我解释一下这个递归函数吗?