C++ 当鼠标位于某个控件上时,如何设置自定义光标?
我想在鼠标位于某个控件上时更改光标。我有我的光标的png。如何在C++中实现? 我就这样试过了 但它说OCR_NORMAL是未定义的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); 我也尝试过这样做,但它会产
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的头文件中定义。假设您的IDIDC\u FUNKY
)
现在,您已经从嵌入EXE的资源中加载了时髦的光标。当然,LoadCursor
也可用于加载预定义的系统游标。要做到这一点,您需要为第一个参数传递NULL
,因为您不是从应用程序的资源加载它,而是从系统加载它。例如,让我们加载帮助光标:
HCURSOR hCursorHelp = ::LoadCursor(NULL, IDC_HELP);
很好,现在我们知道如何加载光标了。除了一件事:我们处理的所有自定义游标都存储为CUR(或ANI)文件。您在问题中提到要从PNG文件加载光标。老实说,我的建议就是不要那样做。使用可以将PNG文件转换为CUR文件的光标创建程序,只需使用CUR文件即可。否则,您将忙于编写一堆毫无意义的代码来加载PNG文件,将其转换为位图,然后将该位图转换为光标。你一开始就会撞到砖墙;没有明显的方法可以使用Win32 API加载PNG图像。您必须使用GDI+、Windows图像组件或可以处理PNG文件的第三方库。完全超出了这个答案的范围。看看,如果你想从这个兔子洞下去。否则,下载一些类似于转换的东西,然后继续你的生活
那么,总而言之,你应该做以下几点:
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
}