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

标签 c++ mfc

我有 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/

相关文章:

c++ - 在其他计算机上打开 exe 需要什么文件 mvcp100d 需要

c++ - 合并排序中的奇怪错误

c++ - 在 MFC 中将 CWnd 指针转换为 CEdit 指针

c++ - OpenGL:VBO删除的奇怪行为?

c++ - 在不使用 auto 的情况下从模板中的迭代器推断出变量的类型

c++ - 如何调试静态初始化问题?

c++ - 帮助在 MFC 中使用 CWinThread

c++ - MFC 对话框属性表使用在 VS 2017 中产生错误,适用于 VS 2013

c++ - 在MFC中,如何让一个按钮连续创建多个对话框实例而不关闭它们?

c++ - 如何通知父 CDialog 子 CWnd 已更新,以便另一个子控件可以响应更新?