不允许工具提示在CMFCRibbonBar上弹出

时间:2020-03-26 08:51:52

标签: c++ mfc

我拥有CMFCRibbonBar的控制权。我需要创建自定义工具提示。我的工具提示来自CMFCToolTipCtrl,效果很好。但是...

当我将鼠标悬停在功能区按钮上时,将显示工具提示。那很棒。但是,当我将鼠标移出按钮时,工具提示将关闭。那不是我想要的。我只需要能够在工具提示上移动鼠标并单击工具提示上的链接。想象一下,这是某种交互式工具提示。我该怎么做?

1 个答案:

答案 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();
}

这不是理想的解决方案,但可以使用。如果有人有任何建议,我会很高兴听到他们的意见:)