我想在鼠标悬停在某个控件上时更改光标。我有光标的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放在项目目录中,希望它正确。
那么,我将如何设置光标?
答案 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文件的第三方库。完全超出了这个答案的范围。如果你想要走下这个兔子洞,请参见here和here。否则,请下载Greenfish Icon Editor之类的内容进行一次转换,然后继续您的生活。
把所有这些放在一起,然后,这就是你应该做的事情:
LoadCursor
函数的代码,从资源加载光标,为您提供HCURSOR
。在应用程序首次启动时,在初始化例程中执行此操作是明智的。缓存返回的句柄,以便在整个应用程序的生命周期中使用它。如果控件存在于对话框中,则可以在WM_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
}
或者,如果您的控件存在于对话框中,则会略有不同:
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
}
最后一点:您在问题中显示链接器错误,这表明您没有正确地告诉链接器在哪里可以找到Windows SDK。所有这些业务都由&#34; Win32应用程序&#34;自动设置。 Visual Studio中的模板。您应该使用它来创建新项目。如果您还没有这样做,则需要进入项目的设置并告诉链接器使用(至少)kernel32.lib
,user32.lib
和gdi32.lib
。否则,链接器将无法找到您尝试调用的Windows API函数。