C++ 当鼠标位于某个控件上时,如何设置自定义光标?

C++ 当鼠标位于某个控件上时,如何设置自定义光标?,c++,windows,winapi,cursor,C++,Windows,Winapi,Cursor,我想在鼠标位于某个控件上时更改光标。我有我的光标的png。如何在C++中实现? 我就这样试过了 但它说OCR_NORMAL是未定义的 HINSTANCE hInst; hInst = GetModuleHandle(NULL); HCURSOR hCurs; hCurs = LoadCursor(hInst, MAKEINTRESOURCE(2)); ::SetSystemCursor(hCurs,OCR_NORMAL); 我也尝试过这样做,但它会产

我想在鼠标位于某个控件上时更改光标。我有我的光标的png。如何在C++中实现?

我就这样试过了

但它说OCR_NORMAL是未定义的

 HINSTANCE hInst;           
 hInst = GetModuleHandle(NULL);

 HCURSOR hCurs;    
 hCurs = LoadCursor(hInst, MAKEINTRESOURCE(2));

::SetSystemCursor(hCurs,OCR_NORMAL); 
我也尝试过这样做,但它会产生奇怪的链接器错误,如:

Error   2   error LNK2019: unresolved external symbol "extern "C" struct HICON__ * __stdcall LoadCursorW(struct HINSTANCE__ *,wchar_t const *)" (?LoadCursorW@@$$J18YGPAUHICON__@@PAUHINSTANCE__@@PB_W@Z) referenced in function "int __cdecl main(void)" (?main@@$$HYAHXZ) C:\Users\Diozz\Documents\Visual Studio 2013\Projects\Scroller\Scroller\main.obj
我将png放在项目目录中,希望它是正确的


那么,如何设置光标呢?

如果要在光标位于特定控件上时更改光标,则需要处理该控件窗口的
WM_SETCURSOR
消息。收到此消息后,您将调用
SetCursor
函数设置应显示的光标。此函数只接受一个参数,即光标的句柄(
HCURSOR
)。关于这方面的更多背景知识,你一定要阅读Raymond Chen的文章

在任何情况下,您都不会调用
SetSystemCursor
函数。该函数为您提供了一种更改全局光标设置的方法,这些设置与您在鼠标控制面板中更改的设置相同。如果用户想要定制自己的桌面,这取决于用户的更改。申请者不应理会这一点。如果您想在应用程序中的控件上显示一个时髦的光标,这是完全可以的,但是如果您用时髦的光标替换系统范围的箭头光标,这是不好的

这样,我们就不必担心调用
SetSystemCursor
的正确方法。让我们看看如何加载游标。您已经找到了
LoadCursorFromFile
函数,实际上,这个函数完全按照它的名字来做。您给它一个CUR文件的路径,它将其作为光标加载,并向您传递该光标的句柄(
HCURSOR
)。但是,除了出于测试目的,您可能不会发现自己曾经使用过
LoadCursorFromFile
。为什么?因为您不想在应用程序中部署CUR文件。如果该文件被删除或未包含,则应用程序将停止工作

相反,光标应该直接链接到应用程序的二进制文件中。幸运的是,Windows作为二进制文件资源的一部分提供了这样做的方法。如果您以前做过任何Windows编程,那么您肯定已经看到了资源文件。对于RC文件,可以添加光标资源,这相当于指定ICO文件的路径。然后,资源编译器执行其余操作,将光标直接嵌入到EXE中。这样,在运行时,您就不必再依赖脆弱的路径,只需调用
LoadCursor
从资源加载光标。(所有资源都有一个数字ID,在名为Resource.h的头文件中定义。假设您的ID
IDC\u FUNKY

现在,您已经从嵌入EXE的资源中加载了时髦的光标。当然,
LoadCursor
也可用于加载预定义的系统游标。要做到这一点,您需要为第一个参数传递
NULL
,因为您不是从应用程序的资源加载它,而是从系统加载它。例如,让我们加载帮助光标:

HCURSOR hCursorHelp = ::LoadCursor(NULL, IDC_HELP);
很好,现在我们知道如何加载光标了。除了一件事:我们处理的所有自定义游标都存储为CUR(或ANI)文件。您在问题中提到要从PNG文件加载光标。老实说,我的建议就是不要那样做。使用可以将PNG文件转换为CUR文件的光标创建程序,只需使用CUR文件即可。否则,您将忙于编写一堆毫无意义的代码来加载PNG文件,将其转换为位图,然后将该位图转换为光标。你一开始就会撞到砖墙;没有明显的方法可以使用Win32 API加载PNG图像。您必须使用GDI+、Windows图像组件或可以处理PNG文件的第三方库。完全超出了这个答案的范围。看看,如果你想从这个兔子洞下去。否则,下载一些类似于转换的东西,然后继续你的生活

那么,总而言之,你应该做以下几点:

  • 将PNG文件转换为ICO文件,并将此ICO文件作为资源添加到应用程序中。您可以在VisualStudio中轻松地执行此操作
  • 编写调用
    LoadCursor
    函数从资源加载光标的代码,为您提供
    HCURSOR
    。在应用程序首次启动时,在初始化例程中这样做是明智的。缓存返回的句柄,以便在应用程序的整个生命周期中使用它。如果控件位于对话框上,则可以在
    WM\u INITDIALOG
    中执行此操作
  • 处理控件的
    WM_SETCURSOR
    消息。尽管您可以通过子类化来实现这一点,但在大多数情况下,只需将代码放入父级的窗口过程中即可:

    static HCURSOR hCursorFunky;
    
    ...
    
    case WM_SETCURSOR:
    {
        // If we're the control that should get the cursor treatment...
        if (static_cast<HWND>(wParam) == hwndYourControl)
        {
            ::SetCursor(hCursorFunky);
            return TRUE;  // indicate we processed this message
        }
        return ::DefWindowProc(hWnd, uMsg, wParam, lParam);  // do default handling
    }
    
    静态HCURSOR hCursorFunky;
    ...
    案例WM_SETCURSOR:
    {
    //如果我们是应该得到光标处理的控件。。。
    if(静态_转换(wParam)=hwndYourControl)
    {
    ::设置光标(hCursorFunky);
    return TRUE;//表示已处理此消息
    }
    return::DefWindowProc(hWnd、uMsg、wParam、lParam);//执行默认处理
    }
    
    或者,如果控件位于对话框中,则有一点变化:

    case WM_SETCURSOR:
    {
        if (static_cast<HWND>(wParam) == ::GetDlgItem(hWnd, IDC_YOURCONTROL))
        {
            ::SetCursor(hCursorFunky);
            ::SetWindowLongPtr(hWnd, DWLP_MSGRESULT, TRUE);
            return TRUE;  // indicate we processed this message
        }
        return FALSE;  // do default handling
    }
    
    case WM_SETCURSOR:
    {
    if(static_cast(wParam)=:::GetDlgItem(hWnd,IDC_YOURCONTROL))
    {
    ::设置光标(hCursorFunky);
    ::SetWindowLongPtr(hWnd,DWLP_MSGRESULT,TRUE);
    return TRUE;//表示已处理此消息
    }
    返回FALSE;//执行默认处理
    }
    

  • 最后一点注意:您的问题中显示了一个链接器错误,这表明您没有正确地告诉链接器在哪里可以找到Windows SDK。所有这些业务都是为您自动设置的
    static HCURSOR hCursorFunky;
    
    ...
    
    case WM_SETCURSOR:
    {
        // If we're the control that should get the cursor treatment...
        if (static_cast<HWND>(wParam) == hwndYourControl)
        {
            ::SetCursor(hCursorFunky);
            return TRUE;  // indicate we processed this message
        }
        return ::DefWindowProc(hWnd, uMsg, wParam, lParam);  // do default handling
    }
    
    case WM_SETCURSOR:
    {
        if (static_cast<HWND>(wParam) == ::GetDlgItem(hWnd, IDC_YOURCONTROL))
        {
            ::SetCursor(hCursorFunky);
            ::SetWindowLongPtr(hWnd, DWLP_MSGRESULT, TRUE);
            return TRUE;  // indicate we processed this message
        }
        return FALSE;  // do default handling
    }