我有 CMFCRibbonBar
控件。我需要创建自定义工具提示。我的工具提示源自 CMFCToolTipCtrl
并且工作得很好。但是...
当我将鼠标悬停在功能区按钮上时,会显示工具提示。那太棒了。但是当我将鼠标移出按钮时,工具提示会关闭。那不是我想要的。我只需要能够将鼠标移动到工具提示上并单击工具提示上的链接。想象一下这是某种交互式工具提示。我可以做什么来实现这一目标?
最佳答案
好吧,我做了一些有用的事情,但结果并没有 100% 令人满意。
因此,首先,创建您自己的工具提示,继承自 CMfcToolTipCtrl。 这个想法是: - 用户可能想与您的工具提示进行交互,也可能不想。因此,我们必须创建一些关闭和显示工具提示的智能方法。 - 我们可以假设,当用户用鼠标悬停工具提示时,他想要进行交互。
不幸的是,每当用户将鼠标从功能区按钮上移开时,工具提示就会消失。但有时我们可以在其中捕获 MouseMove。但这种情况相当罕见。因此,我们必须捕获系统关闭工具提示的时刻。
有这样一条消息,我们可以添加到消息映射中:
ON_NOTIFY_REFLECT(TTN_POP, &CAsInteractiveToolTip::OnPop)
现在,我们的 OnPop 看起来像这样(我正在使用 pImpl 习惯用法):
void CAsInteractiveToolTip::OnPop(NMHDR* pNMHDR, LRESULT* pResult)
{
if (m_pImpl->m_forceClose)
{
CMFCToolTipCtrl::OnPop(pNMHDR, pResult);
m_pImpl->m_forceOpened = false;
m_pImpl->m_forceClose = false;
m_pImpl->StopForceOpenTimer();
}
else
{
m_pImpl->StartForceOpenTimer();
}
*pResult = 0;
}
现在,这里发生的是: - 当工具提示关闭时,检查它是否被我们的代码强制关闭。如果没有的话,说明它被系统关闭了。在这种情况下,我们必须让用户有机会将鼠标悬停在我们的工具提示上。因此,我们必须再次显示工具提示(强制显示)。这是在计时器方法中完成的。 StartForceOpenTimer 是启动计时器的简单方法:
void StartForceOpenTimer()
{
if (!m_forceOpenTimerActive)
{
m_self.SetTimer(IDT_FORCE_OPEN_TIMER, 100, (TIMERPROC)NULL);
m_forceOpenTimerActive = true;
}
}
现在,魔法从计时器方法开始:
void CAsInteractiveToolTip::OnForceTimer()
{
static DWORD waitForUserStartTime = 0;
static bool waitingForUserReaction = false;
if (!waitingForUserReaction)
{
//open and give the user chance to mouse move over it within 0.5 seconds
SetWindowPos(&wndTopMost, -1, -1, -1, -1, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_SHOWWINDOW);
waitForUserStartTime = GetTickCount();
waitingForUserReaction = true;
return;
}
if (GetTickCount() - waitForUserStartTime > 500)
{
m_pImpl->StopForceOpenTimer();
m_pImpl->m_forceClose = true;
waitingForUserReaction = false;
m_pImpl->PopToolTip();
return;
}
if (m_pImpl->m_doForceOpen)
{
m_pImpl->StopForceOpenTimer();
waitingForUserReaction = false;
m_pImpl->m_forceOpened = true;
}
}
总体思路是: - 强制显示工具提示 - 等待用户悬停鼠标约 0.5 秒 - 如果用户将鼠标悬停在工具提示窗口上,我们可以假设他想要进行交互。所以我们可以让 window 开着。 - 如果用户在 0.5 秒内没有与窗口交互,只需再次关闭工具提示即可。
现在,PopToolTip 方法只是启动另一个计时器,间隔为 100 毫秒。 这是魔法的另一部分:
void CAsInteractiveToolTip::OnPopTimer()
{
m_pImpl->StopForceOpenTimer();
KillTimer(IDT_POP_TIMER);
//Pop();
m_pImpl->m_forceClose = true;
m_pImpl->m_hdr.idFrom = 2;
m_pImpl->m_hdr.hwndFrom = GetSafeHwnd();
m_pImpl->m_hdr.code = (int)TTN_POP; //4294966774
GetParent()->SendMessage(WM_NOTIFY, 1, (LPARAM)&m_pImpl->m_hdr);
//GetParent()->SendMessage(WM_NOTIFY, 2, (LPARAM)&m_pImpl->m_hdr);
ShowWindow(SW_HIDE);
}
现在,此方法应该只是弹出(隐藏)工具提示。但由于某种原因,在我的例子中,调用 Pop() 方法没有任何作用。所以我必须发送带有适当参数的 WM_NOTIFY 消息(它们取 self 的调试观察)。 现在,OnPop 将再次启动,但这次 m_forceClose 设置为 true,因此工具提示将不会再次显示(第一个计时器将不会运行)。
现在是魔法的第三部分——鼠标移动。只需将其添加到您的消息映射中即可:
ON_WM_MOUSEMOVE()
方法:
void CAsInteractiveToolTip::OnMouseMove(UINT nFlags, CPoint point)
{
m_pImpl->m_doForceOpen = true; //let the first timer know, that user wants to interact
CMFCToolTipCtrl::OnMouseMove(nFlags, point);
}
当用户单击工具提示时,您可以隐藏它。只是:
void CAsInteractiveToolTip::OnLButtonDown(UINT nFlags, CPoint point)
{
m_pImpl->m_forceClose = true;
m_pImpl->PopToolTip();
}
这不是理想的解决方案,但它确实有效。如果有人有任何建议,我会很高兴听到他们:)
关于c++ - 不允许在 CMFCRibbonBar 上弹出工具提示,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60863683/