我拥有CMFCRibbonBar
的控制权。我需要创建自定义工具提示。我的工具提示来自CMFCToolTipCtrl
,效果很好。但是...
当我将鼠标悬停在功能区按钮上时,将显示工具提示。那很棒。但是,当我将鼠标移出按钮时,工具提示将关闭。那不是我想要的。我只需要能够在工具提示上移动鼠标并单击工具提示上的链接。想象一下,这是某种交互式工具提示。我该怎么做?
答案 0 :(得分:0)
好的,我做了一些有用的事情,但结果并不令人满意。
因此,首先,从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秒,让用户将鼠标悬停 -如果用户将鼠标悬停在工具提示窗口上,则可以假定他要进行交互。这样我们就可以打开窗口了。 -如果用户在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消息(它们来自我的调试观察)。 现在,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();
}
这不是理想的解决方案,但可以使用。如果有人有任何建议,我会很高兴听到他们的意见:)