通过


使用 Visual C++ 更改 MFC 中窗口的鼠标指针

本文介绍如何使用 Visual C++ 更改 MFC 中窗口的鼠标指针。 本文中的信息仅适用于非托管的 Visual C++ 代码。

原始产品版本: Visual C++
原始 KB 数: 131991

总结

在基于 Windows 的应用程序中,始终基于窗口类创建窗口。 窗口类基于其定义的特性识别多个窗口特征,包括默认鼠标指针(光标)。 在某些情况下,应用程序可能需要更改与它创建的某些窗口关联的指针。 本文介绍 MFC 应用程序可用于在不同时间显示不同指针的三种方法。

MFC 应用程序显示不同指针的情况

在一些情况下,你可能希望 MFC 应用程序在不同时间显示不同的指针:

  • 当默认指针不是特定应用程序的好用户界面对象时。 例如,I-beam 指针比记事本中的文本编辑器窗口的箭头更合适。 这可能涉及更改应用程序整个运行过程的指针。
  • 当应用程序执行长时间的操作(如磁盘 I/O)时,沙漏指针比箭头更合适。 通过将光标更改为沙漏形状,可以为用户提供良好的视觉反馈。 这可能涉及在有限的时间段内更改指针。

在窗口中更改鼠标指针的三种方法

下面是应用程序可以在窗口中更改鼠标指针的三种方式:

  • 方法 1:重写 CWnd::OnSetCursor() 函数。 调用 Windows API SetCursor() 函数以更改指针。
  • 方法 2:使用所需的鼠标指针注册自己的窗口类,重写 CWnd::PreCreateWindow() 函数,并使用新注册的窗口类创建窗口。
  • 方法 3:若要显示标准沙漏指针,应用程序可以调用 CCmdTarget::BeginWaitCursor()显示沙漏,并调用 CmdTarget::EndWaitCursor() 以恢复为默认指针。 此方案仅适用于单条消息的时间段。 如果在调用 EndWaitCursor 之前移动鼠标,Windows 会将消息发送到 WM_SETCURSOR 指针下方的窗口。 此消息的默认处理会将指针重置为类中注册的默认类型,因此您需要为该窗口重写 CWnd::OnSetCursor(),并将指针重置为沙漏。

以下代码示例通过示例演示如何使用三种方法更改派生类窗口的 CView 鼠标指针。

m_ChangeCursor 是类的成员 CMyView 变量,属于类型 BOOL。 它指示是否需要显示其他指针类型。

方法 1 的代码

通过重写CWnd::OnSetCursor()函数,更改CMyView对象的鼠标指针。 使用类向导为 Windows 消息CMyView::OnSetCursor()建立消息映射函数WM_SETCURSOR,并提供函数正文,如下所示:

BOOL CMyView::OnSetCursor(CWnd *pWnd, UINT nHitTest, UINT message)
{
    if (m_ChangeCursor)
    {
        ::SetCursor(AfxGetApp()->LoadStandardCursor(IDC_WAIT));
        return TRUE;
    }
    return CView::OnSetCursor(pWnd, nHitTest, message);
}

方法 2 的代码

使用 AfxRegisterClass()AfxRegisterWndClass() 函数注册自己的窗口类,其中包含所需的鼠标指针。 然后基于已注册的窗口类创建视图窗口。 有关在 MFC 中注册窗口类的详细信息,请参阅 MFC 技术说明 1 中的 窗口类注册

BOOL CMyView::PreCreateWindow(CREATESTRUCT &cs)
{
    cs.lpszClass = AfxRegisterWndClass(
        CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW, // use any window styles
        AfxGetApp()->LoadStandardCursor(IDC_WAIT),
        (HBRUSH)(COLOR_WINDOW + 1)); // background brush
    return CView::PreCreateWindow(cs)
}

方法 3 的代码

调用BeginWaitCursor()EndWaitCursor()函数以更改鼠标指针。

注释

CWinApp::DoWaitCursor(1)CWinApp::DoWaitCursor(-1)的工作方式与BeginWaitCursor()EndWaitCursor()类似。

void CMyView::PerformLengthyOperation()
{
    BeginWaitCursor(); // or AfxGetApp()->DoWaitCursor(1)
    //...
    EndWaitCursor(); // or AfxGetApp()->DoWaitCursor(-1)
}

如果 BeginWaitCursor()EndWaitCursor() 的调用不在同一个处理程序中,则必须按如下所示重写 OnSetCursor

BOOL CMyView::OnSetCursor(CWnd *pWnd, UINT nHitTest, UINT message)
{
    if (m_ChangeCursor)
    {
        RestoreWaitCursor();
        return TRUE;
    }
    return CView::OnSetCursor(pWnd, nHitTest, message);
}

在此示例中,在调用m_ChangeCursor前设置为 BeginWaitCursor(),并在调用后将其设置为 EndWaitCursor()