编辑:我忘了提,我没有创建窗口的DLL的源代码,所以我实际上不能改变函数返回HWND。
我正在创建一个Win32应用程序,我正在使用一个DLL,通过它的一个导出函数"void X()"为我创建一个窗口;我在WinMain()中调用X()
它确实为我创建了一个窗口。我想获得这个导出的库函数创建的窗口的HWND,因为X()返回void,所以我可以将它用于其他API调用。有人能告诉我去HWND最容易吗?
我已经在这里搜索并回答了问题,但我无法找到确切的,适当的解决方案。我尝试了EnumWIndows(),然后获得进程ID,然后与当前线程进程ID进行比较。但我想应该有一种更好、更有效、更简单的方法来获得HWND。毕竟,我是在创建这个窗口的进程的WinMain中。
如果我需要解释什么,我错过了写在这里,请告诉我。
我确信这是非常基本的,我在这里明显遗漏了一些东西。对不起。谢谢,的问候!
使用spy++或Winspector等工具查看应用程序创建的所有HWND
,特别是它们的类名和窗口标题。然后,您可以将这些值复制到您的代码中,并在DLL创建其窗口后对FindWindow()
进行单个调用,例如:
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
// ...
X();
HWND hWnd = FindWindow("ClassNameHere", "TitleHere");
// ...
return 0;
}
最简单的方法是使用SetWindowsHookEx(WH_CBT, fun, NULL, GetCurrentThreadId())
函数。然后,当许多事件发生时,将调用您定义的回调函数fun
。你想要的是HCBT_CREATEWND
。
有人这样认为(完全未经测试):
HWND hDllHandle = NULL;
LRESULT CALLBACK X_CBTProc(int nCode, WPARAM wParam, LPARAM lParam)
{
if (nCode == HCBT_CREATEWND)
hDllHandle = (HWND)wParam;
return CallNextHookEx(NULL, nCode, wParam, lParam); //The first parameter is useless
}
HWND CallXAndGetHWND()
{
HHOOK hDllHook = SetWindowsHookEx(WH_CBT, X_CBTProc, NULL, GetCurrentThreadId());
X();
UnhookWindowsHookEx(hDllHook);
//hDllHandle is a global variable, so will be now you window!
return hDllHandle;
}
注意,这个函数不是线程感知的,但很可能你只会在代码的开头调用它一次,所以它应该无关紧要。
,小心!许多函数,甚至是Win32 API函数,都会创建隐藏窗口。这段代码将钩住所有的对象,并返回最后一个要创建的对象。如果需要,将其更改为返回任何其他的,甚至是一个列表,应该是微不足道的。