Thinking and Researching

数字冰块的思考空间

  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理
最近在写一个 Alt-Tab 替换程序,需要枚举可切换的应用程序窗口。枚举窗口一般使用的是 EnumWindows 函数
BOOL EnumWindows(
       WNDENUMPROC lpEnumFunc,
    LPARAM lParam
);
但是光是使用这个函数并不能得到我们最需要的窗口,它枚举出的窗口有点“多”,包含隐藏的,包含带有子窗体属性的,包含只出现在任务栏通知区域的。。。等等。因而需要有一些过滤条件。我在 Google 上找了很久都没有一个满意的答案,只能自己用 Spy++ 观察一些窗口做了个过滤规则,经过过滤的就是“可显示、可激活、可切换”的窗口。过滤规则请见代码注释。
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;
..........
..........略
posted on 2006-01-26 00:53  数字冰块  阅读(7232)  评论(3编辑  收藏  举报