当鼠标悬停在某个控件上时,如何设置自定义光标?

时间:2016-06-17 16:55:59

标签: c++ windows winapi cursor

我想在鼠标悬停在某个控件上时更改光标。我有光标的png。我如何用C ++实现它?

我按照here

所述尝试过
HCURSOR hcur;
hcur = ::LoadCursorFromFile("cursor.png");
::SetSystemCursor(hcur,OCR_NORMAL);

但它说 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放在项目目录中,希望它正确。

那么,我将如何设置光标?

1 个答案:

答案 0 :(得分:3)

如果要在特定控件上更改光标,则需要处理该控件窗口的WM_SETCURSOR消息。收到此消息后,您将调用SetCursor函数来设置应显示的光标。此函数采用单个参数,即光标的句柄(HCURSOR)。有关这方面的更多背景知识,您一定要阅读Raymond Chen的文章"What is the process by which the cursor gets set?"

在任何情况下,您都不会调用SetSystemCursor函数。该功能为您提供了一种更改全局光标设置的方法 - 您知道,您在鼠标控制面板中更改了相同的设置。如果她想自定义她的桌面,则由用户进行更改。应用程序应该单独留下。如果你想在你的应用程序中的控件上显示一个时髦的光标,那就完全没了问题,但如果你用一个时髦的箭头光标替换系统范围的箭头光标,那么就好了!

如果不这样做,我们真的不必担心拨打SetSystemCursor的正确方法。所以让我们来看看加载游标。您已经找到了LoadCursorFromFile函数,实际上,这个函数正如其名称所暗示的那样。你给它一个CUR文件的路径,然后把它作为游标加载,传递给那个游标的句柄(HCURSOR)。但是,除了测试目的之外,您可能还没有发现自己曾经使用LoadCursorFromFile。为什么?因为您不希望必须与应用程序一起部署CUR文件。如果该文件被删除或未包含,则您的应用程序将停止工作。

相反,游标应直接链接到应用程序的二进制文件中。幸运的是,Windows提供了一种方法,可以将其作为二进制文件资源的一部分。如果您之前已经完成过任何Windows编程,那么您肯定会看到资源文件。对于RC文件,您可以添加游标资源,这相当于指定ICO文件的路径。然后资源编译器完成剩下的工作,将光标直接嵌入到EXE中。完成后,在运行时,您不再需要依赖脆弱的路径,只需调用LoadCursor从资源加载光标即可。 (所有资源都有一个数字ID,在名为Resource.h的头文件中定义。让我们假设您的ID为IDC_FUNKY。)

HINSTANCE hInstance = ::GetModuleHandle(NULL);  // get a handle to the app's instance
HCURSOR   hCursor   = ::LoadCursor(hInstance, MAKEINTRESOURCE(IDC_FUNKY));

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

HCURSOR hCursorHelp = ::LoadCursor(NULL, IDC_HELP);

很好 - 现在我们知道如何加载游标。除了一件事之外:我们处理过的所有自定义光标都存储为CUR(或ANI)文件。您在问题中提到要从PNG文件加载游标。老实说,我的建议只是来做到这一点。使用可以将PNG文件转换为CUR文件的光标创建程序,只需使用CUR文件即可。否则,您将忙于编写一堆无意义的代码来加载PNG文件,将其转换为位图,然后将该位图转换为游标。一旦你开始,你就会碰到一堵砖墙;没有明显的方法可以使用Win32 API加载PNG图像。您必须使用GDI +,Windows Imaging Component或可以处理PNG文件的第三方库。完全超出了这个答案的范围。如果你想要走下这个兔子洞,请参见herehere。否则,请下载Greenfish Icon Editor之类的内容进行一次转换,然后继续您的生活。

把所有这些放在一起,然后,这就是你应该做的事情:

  1. 将PNG文件转换为ICO文件,并将此ICO文件作为资源添加到应用程序中。您可以在Visual Studio中轻松完成此操作。
  2. 编写调用LoadCursor函数的代码,从资源加载光标,为您提供HCURSOR。在应用程序首次启动时,在初始化例程中执行此操作是明智的。缓存返回的句柄,以便在整个应用程序的生命周期中使用它。如果控件存在于对话框中,则可以在WM_INITDIALOG
  3. 中执行此操作
  4. 处理控件的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
    }
    

    或者,如果您的控件存在于对话框中,则会略有不同:

    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
    }
    
  5. 最后一点:您在问题中显示链接器错误,这表明您没有正确地告诉链接器在哪里可以找到Windows SDK。所有这些业务都由&#34; Win32应用程序&#34;自动设置。 Visual Studio中的模板。您应该使用它来创建新项目。如果您还没有这样做,则需要进入项目的设置并告诉链接器使用(至少)kernel32.libuser32.libgdi32.lib。否则,链接器将无法找到您尝试调用的Windows API函数。