gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-02 10:33:16 25 4
gpt4 key购买 nike

除了绘制文本之外,我仍然在调整 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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com