0

我正在尝试在 MFC 对话框上设置工具提示。我有从字符串资源读取工具提示的代码,并试图将其修改为不从资源中读取,而是组成一个工具提示。

我的实现导致垃圾显示为工具提示,而不是我想要的字符串。此外,从调试器运行时会导致崩溃,但在直接运行可执行文件时不会导致崩溃(我确信存在缓冲区损坏或类似情况)

这是相关的代码:

BOOL CPreviewDlg::OnToolTipText(UINT, NMHDR* pNMHDR, LRESULT* pResult)
{
    ASSERT(pNMHDR->code == TTN_NEEDTEXT);

    TOOLTIPTEXT* pTTT = (TOOLTIPTEXT*)pNMHDR;

    if (!(pTTT->uFlags & TTF_IDISHWND))
        return FALSE;

    UINT_PTR hWnd = pNMHDR->idFrom;
    // idFrom is actually the HWND of the tool
    UINT nID = (UINT)(WORD)::GetDlgCtrlID((HWND)hWnd);

    CString sDlgItemText;
    (UINT)(WORD)::GetDlgItemText(this->GetSafeHwnd(), nID, sDlgItemText.GetBufferSetLength(50), 50);
    sDlgItemText.ReleaseBuffer();

    if(sDlgItemText.IsEmpty())
        sDlgItemText = _T("Unnamed");

    CString sToolTip = _T("");
    sToolTip.Format(_T("%s \n This is the %s control. Here we can put its description."), sDlgItemText, sDlgItemText);

    pTTT->lpszText = sToolTip.GetBufferSetLength(sToolTip.GetLength());/* MAKEINTRESOURCE(nID);*/
    pTTT->hinst = AfxGetInstanceHandle();
    sToolTip.ReleaseBuffer();
    *pResult = 0;

    // bring the tooltip window above other popup windows
    ::SetWindowPos(pNMHDR->hwndFrom, HWND_TOP, 0, 0, 0, 0,
        SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOOWNERZORDER);

    return TRUE;    // message was handled
}

我很确定错误是我试图设置的地方pTTT->lpszText(而不是 MAKEINTRESOURCE)我认为我没有得到从 CString 设置 LPSTR 的正确方法

CString sDlgItemText;
(UINT)(WORD)::GetDlgItemText(this->GetSafeHwnd(), nID, sDlgItemText.GetBufferSetLength(50), 50);
sDlgItemText.ReleaseBuffer();

if(sDlgItemText.IsEmpty())
    sDlgItemText = _T("Unnamed");

CString sToolTip = _T("");
sToolTip.Format(_T("%s \n This is the %s control. Here we can put its description."), sDlgItemText, sDlgItemText);

pTTT->lpszText = sToolTip.GetBufferSetLength(sToolTip.GetLength());/* MAKEINTRESOURCE(nID);*/
4

2 回答 2

1

我认为你的问题是pTTT->lpszText一旦你退出函数,指针值就不再有效。

如果您的文本长度少于 80 个字符,或者您有能力截断工具提示,则可以使用

lstrcpy(pTTT->szText, sToolTip);

否则,您将需要创建sToolTip一个全局变量或CPreviewDlg类的成员变量。

参考:MSDN 文章

于 2013-09-19T14:15:02.407 回答
0

根本问题是您正在返回局部变量的地址(sToolTip.m_pData)从您的OnToolTipText处理程序。当控制离开通知处理程序时,sToolTip超出范围并且其析构函数运行,留下垃圾。

要解决此问题,您有 2 个选项:

  1. 将工具提示文本复制到TOOLTIPTEXT::szText[]数组中。
  2. 增加您返回的缓冲区的生命周期,以便在系统需要时可用。生命周期不需要比封闭实体(本例中的对话框)长,因此对话框的类成员就可以了。

与您的问题无关:将指针分配给TOOLTIPTEXT::lpszText成员时,您应该使用 aconst_cast而不是调用CString::GetBuffer[SetLength](). 该TOOLTIPTEXT结构用于两个方向,以设置和检索工具提示信息。因此,成员不能被声明const,即使它们是。它可能看起来很尴尬,但您更愿意执行以下操作:

pTTT->lpszText = const_cast<LPTSTR>(static_cast<LPCTSTR>(sToolTip));
于 2013-09-19T15:46:15.893 回答