最近在写一个 Alt-Tab 替换程序,需要枚举可切换的应用程序窗口。枚举窗口一般使用的是 EnumWindows 函数
BOOL EnumWindows(
WNDENUMPROC lpEnumFunc,
LPARAM lParam
);
但是光是使用这个函数并不能得到我们最需要的窗口,它枚举出的窗口有点“多”,包含隐藏的,包含带有子窗体属性的,包含只出现在任务栏通知区域的。。。等等。因而需要有一些过滤条件。我在 Google 上找了很久都没有一个满意的答案,只能自己用 Spy++ 观察一些窗口做了个过滤规则,经过过滤的就是“可显示、可激活、可切换”的窗口。过滤规则请见代码注释。
WNDENUMPROC lpEnumFunc,
LPARAM lParam
);
BOOL CALLBACK EnumWindowsProc(HWND hwnd,LPARAM lParam)
{
//窗口是否可视
if (!IsWindowVisible(hwnd))
return TRUE;
//窗口是否可激活
if (!IsWindowEnabled(hwnd))
return TRUE;
//窗口是否 WS_POPUP 与 WS_CAPTION 共存
//一些可切换的窗体同时具有 WS_POPUP 与 WS_CAPTION,因而有 WS_POPUP 却无 WS_CAPTION 的应被过滤
//据 Spy++ 观察,符合如 OneNote TrayIcon 等程序可通过此方式过滤
LONG gwl_style = GetWindowLong(hwnd,GWL_STYLE);
if ((gwl_style & WS_POPUP) && !(gwl_style & WS_CAPTION))
return TRUE;
//窗口是否具有父窗口?
HWND hParent = (HWND)GetWindowLong(hwnd,GWL_HWNDPARENT);
//父窗口是否可激活?
//据 Spy++ 观察,如“运行”对话框等被应列入列表的程序有一个隐藏的,具有 WS_DISABLED 的父窗口
if (IsWindowEnabled(hParent))
return TRUE;
//父窗口是否可视?
if (IsWindowVisible(hParent))
return TRUE;
//一个非常奇怪的问题在于,任务栏 Shell_TrayWnd 符合上述过滤条件但是无法被过滤。
//因而在这里单独列出。
TCHAR szClassName[MAX_LOADSTRING];
GetClassName(hwnd,szClassName,MAX_LOADSTRING);
if (!wcscmp(szClassName,L"Shell_TrayWnd"))
return TRUE;
..........
..........略
{
//窗口是否可视
if (!IsWindowVisible(hwnd))
return TRUE;
//窗口是否可激活
if (!IsWindowEnabled(hwnd))
return TRUE;
//窗口是否 WS_POPUP 与 WS_CAPTION 共存
//一些可切换的窗体同时具有 WS_POPUP 与 WS_CAPTION,因而有 WS_POPUP 却无 WS_CAPTION 的应被过滤
//据 Spy++ 观察,符合如 OneNote TrayIcon 等程序可通过此方式过滤
LONG gwl_style = GetWindowLong(hwnd,GWL_STYLE);
if ((gwl_style & WS_POPUP) && !(gwl_style & WS_CAPTION))
return TRUE;
//窗口是否具有父窗口?
HWND hParent = (HWND)GetWindowLong(hwnd,GWL_HWNDPARENT);
//父窗口是否可激活?
//据 Spy++ 观察,如“运行”对话框等被应列入列表的程序有一个隐藏的,具有 WS_DISABLED 的父窗口
if (IsWindowEnabled(hParent))
return TRUE;
//父窗口是否可视?
if (IsWindowVisible(hParent))
return TRUE;
//一个非常奇怪的问题在于,任务栏 Shell_TrayWnd 符合上述过滤条件但是无法被过滤。
//因而在这里单独列出。
TCHAR szClassName[MAX_LOADSTRING];
GetClassName(hwnd,szClassName,MAX_LOADSTRING);
if (!wcscmp(szClassName,L"Shell_TrayWnd"))
return TRUE;
..........
..........略