mat*_*son 4 c++ events alt-tab windows-10 visual-studio-2017
添加信息:
当我在 Windows 7 中使用 .exe 文件时,一切都会正常工作。它在 Windows 10 中不起作用。我不明白为什么。
原帖:
也许我只是需要更多咖啡......但我正在尝试编写一个小程序来监听 Alt+Tab 事件。因此,我添加了SetWinEventHook来监听EVENT_SYSTEM_SWITCHSTART和EVENT_SYSTEM_SWITCHEND。因为这就是我认为根据 MSDN 执行此操作的方法:SetWinEventHook、WinEventProc 回调、Event 常量
由于某种原因,这些事件似乎从未发生过。但更可能的是我错过了一些东西。
当我没有让它在我的实际应用程序中工作时,我创建了这个小应用程序来测试它。它也在那里不起作用。但其他事件确实有效。EVENT_SYSTEM_FOREGROUND、EVENT_OBJECT_CREATE和 EVENT_OBJECT_DESTROY一直在发生。
我的小型测试应用程序的代码:
#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
namespace
{
HWINEVENTHOOK sTabHook;
HWINEVENTHOOK sFocusHook;
HWINEVENTHOOK sCreateHook;
}
void CALLBACK tabEventProc(HWINEVENTHOOK hook, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread, DWORD dwmsEventTime)
{
printf("Never happens!");
if (EVENT_SYSTEM_SWITCHSTART == event)
{
printf("Tab start\n");
}
else if (EVENT_SYSTEM_SWITCHEND == event)
{
printf("Tab end\n");
}
}
void CALLBACK focusEventProc(HWINEVENTHOOK hook, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread, DWORD dwmsEventTime)
{
printf("Window focused\n");
}
void CALLBACK createEventProc(HWINEVENTHOOK hook, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread, DWORD dwmsEventTime)
{
if (EVENT_OBJECT_CREATE == event)
{
printf("Object created\n");
}
else if (EVENT_OBJECT_DESTROY == event)
{
printf("Object destroyed\n");
}
}
DWORD WINAPI threadProc()
{
sTabHook = SetWinEventHook(EVENT_SYSTEM_SWITCHSTART, EVENT_SYSTEM_SWITCHEND, nullptr, tabEventProc, 0, 0, WINEVENT_OUTOFCONTEXT | WINEVENT_SKIPOWNPROCESS); //no ice, no napkins, no soda, no salt, no pepper... no quaso, NOTHING!
sFocusHook = SetWinEventHook(EVENT_SYSTEM_FOREGROUND, EVENT_SYSTEM_FOREGROUND, nullptr, focusEventProc, 0, 0, WINEVENT_OUTOFCONTEXT | WINEVENT_SKIPOWNPROCESS);
sCreateHook = SetWinEventHook(EVENT_OBJECT_CREATE, EVENT_OBJECT_DESTROY, nullptr, createEventProc, 0, 0, WINEVENT_OUTOFCONTEXT | WINEVENT_SKIPOWNPROCESS);
MSG message;
while (GetMessage(&message, nullptr, 0, 0)) {
TranslateMessage(&message);
DispatchMessage(&message);
}
UnhookWinEvent(sTabHook);
UnhookWinEvent(sFocusHook);
UnhookWinEvent(sCreateHook);
return 0;
}
int main()
{
HANDLE threadHandle;
DWORD thread;
threadHandle = CreateThread(nullptr, 0, (LPTHREAD_START_ROUTINE)threadProc, 0, 0, &thread);
if (threadHandle)
{
return WaitForSingleObject(threadHandle, INFINITE);
}
else
{
return 1;
}
return 0;
}
Run Code Online (Sandbox Code Playgroud)
单击窗口和打开/关闭窗口时,我从focusEventProc和createEventProc方法获得大量打印输出。但是当 alt+tab 键切换到其他应用程序时, tabEventProc没有打印输出。
如果我在方法中放置断点,也会发生同样的情况。它永远不会在tabEventProc内部中断,但始终在其他两个方法中中断。
有人可以告诉我我做错了什么吗?因为 Google Wan Kenobi 无法帮助我。我真的很困惑。
我使用的是 Windows 10 和 Visual Studio 2017 社区版。该项目是从 Visual Studios“创建新项目”模板创建的标准 C++ 控制台项目。
好的。所以我想你必须遵守丑陋的黑客才能使 Alt-Tab 检测在 Windows 10 上工作。
这是我最终得到的丑陋的解决方案:
#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
#include <string>
namespace
{
HINSTANCE sInstance = GetModuleHandle(nullptr);
HHOOK sTabHook;
HWINEVENTHOOK sFocusHook;
bool sAltTabHappend = false;
}
LRESULT CALLBACK tabEventProc(int nCode, WPARAM wParam, LPARAM lParam)
{
auto kbdLlHookStruct = (KBDLLHOOKSTRUCT*)lParam;
switch (nCode)
{
case HC_ACTION:
{
if (kbdLlHookStruct->vkCode == VK_TAB && kbdLlHookStruct->flags & LLKHF_ALTDOWN)
{
printf("Alt-Tab\n");
sAltTabHappend = true;
}
}
}
return CallNextHookEx(sTabHook, nCode, wParam, lParam);
}
void CALLBACK focusEventProc(HWINEVENTHOOK hook, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread, DWORD dwmsEventTime)
{
if (!sAltTabHappend)
{
return;
}
char windowTitleOut[256];
GetWindowTextA(hwnd, windowTitleOut, sizeof(windowTitleOut));
std::string windowTitle(windowTitleOut);
if (windowTitle.empty() || 0 == windowTitle.compare("Task Switching"))
{
return;
}
printf("You just canged to %s by Alt-tabbing... yeah...\n", windowTitleOut);
sAltTabHappend = false;
}
DWORD WINAPI threadProc()
{
sTabHook = SetWindowsHookEx(WH_KEYBOARD_LL, tabEventProc, sInstance, 0);
sFocusHook = SetWinEventHook(EVENT_SYSTEM_FOREGROUND, EVENT_SYSTEM_FOREGROUND, nullptr, focusEventProc, 0, 0, WINEVENT_OUTOFCONTEXT | WINEVENT_SKIPOWNPROCESS);
MSG message;
while (GetMessage(&message, nullptr, 0, 0)) {
TranslateMessage(&message);
DispatchMessage(&message);
}
UnhookWindowsHookEx(sTabHook);
UnhookWinEvent(sFocusHook);
return 0;
}
int main()
{
HANDLE threadHandle;
DWORD thread;
threadHandle = CreateThread(nullptr, 0, (LPTHREAD_START_ROUTINE)threadProc, 0, 0, &thread);
if (threadHandle)
{
return WaitForSingleObject(threadHandle, INFINITE);
}
else
{
return 1;
}
return 0;
}
Run Code Online (Sandbox Code Playgroud)
因此,我没有连接到EVENT_SYSTEM_SWITCHSTART和EVENT_SYSTEM_SWITCHEND ,而是使用SetWindowsHookEx连接到WH_KEYBOARD_LL。然后我检查 Alt+Tab 按键组合。如果发生这种情况,我会将其保存到静态变量中,并在EVENT_SYSTEM_FOREGROUND处检查 Alt+Tab 之前是否发生过,然后假设我们 Alt+Tab 切换到了该窗口。
我真的很讨厌这个解决方案,因为它不能保证聚焦的窗口是我们切换到的窗口。必须有更好的方法。
它没有回答为什么EVENT_SYSTEM_SWITCHSTART和EVENT_SYSTEM_SWITCHEND在 Windows 10 上不起作用的问题。
归档时间: |
|
查看次数: |
901 次 |
最近记录: |