目前,我可以使用System.Diagnostics.Process.GetProcesses()
获取一个正在运行的进程列表,并执行一个简单的LINQ查询。
然后,我可以导入user32.dll
和SetWindowPos
函数,并操纵其他进程的窗口参数。
好的,它有效。现在我想通过点击它来选择一个进程的窗口,让我们说calc.exe。换句话说,我想获得一个Process对象(然后是MainWindowHandle),它带有一个钩子,当我点击它的窗口时它会捕获进程名称。
我怎样才能做到这一点?
答案 0 :(得分:5)
我不知道在C#中如何做到这一点,但你也标记了这个问题WinAPI,所以我可以帮助那里。在WinAPI中,它可以这样完成:
#include <stdio.h>
#include <Windows.h>
#include <Psapi.h>
#pragma comment(lib, "Psapi.lib")
int main(void)
{
/* Hacky loop for proof of concept */
while(TRUE) {
Sleep(100);
if(GetAsyncKeyState(VK_F12)) {
break;
}
if(GetAsyncKeyState(VK_LBUTTON)) {
HWND hwndPt;
POINT pt;
if(!GetCursorPos(&pt)) {
wprintf(L"GetCursorPos failed with %d\n", GetLastError());
break;
}
if((hwndPt = WindowFromPoint(pt)) != NULL) {
DWORD dwPID;
HANDLE hProcess;
GetWindowThreadProcessId(hwndPt, &dwPID);
hProcess = OpenProcess(PROCESS_QUERY_LIMITED_INFORMATION, FALSE, dwPID);
if(hProcess == NULL) {
wprintf(L"OpenProcess failed with error: %d\n", GetLastError());
} else {
wchar_t lpFileName[MAX_PATH];
DWORD dwSize = _countof(lpFileName);
QueryFullProcessImageName(hProcess, 0, lpFileName, &dwSize);
wprintf(L"%s\n", lpFileName);
CloseHandle(hProcess);
}
}
}
}
return EXIT_SUCCESS;
}
示例结果:
在这种情况下,我只是轮询鼠标点击。更合适的方法是使用某种窗口挂钩。
答案 1 :(得分:1)
正如Mike Kwan所说,虽然两种方法都有自己的缺点,但你最好写一个钩子,但是bjarneds已经做了很好的工作。看看@ DotNET Object Spy。它是用C#编写的,可以满足您的需求。
你还应该注意到,使用挂钩在白天变得多余。根据您的目的,GetForegroundWindow
等其他winapis可能会更好。