在 Windows 应用开发中,如果需要操作其他的窗口,那么可以使用 EnumWindows
这个 API 来枚举这些窗口。
本文介绍使用 EnumWindows
来枚举并找到自己关心的窗口(如 QQ/TIM 窗口)。
EnumWindows
你可以在微软官网了解到 EnumWindows
。
要在 C# 代码中使用 EnumWindows
,你需要编写平台调用 P/Invoke 代码。使用我在另一篇博客中的方法可以自动生成这样的平台调用代码:
我这里直接贴出来:
1
2
[DllImport("user32.dll")]
public static extern int EnumWindows(WndEnumProc lpEnumFunc, int lParam);
遍历所有的顶层窗口
官方文档对此 API 的描述是:
Enumerates all top-level windows on the screen by passing the handle to each window, in turn, to an application-defined callback function.
遍历屏幕上所有的顶层窗口,然后给回调函数传入每个遍历窗口的句柄。
不过,并不是所有遍历的窗口都是顶层窗口,有一些非顶级系统窗口也会遍历到,详见:EnumWindows 中的备注节。
所以,如果需要遍历得到所有窗口的集合,那么可以使用如下代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
public static IReadOnlyList<int> EnumWindows()
{
var windowList = new List<int>();
EnumWindows(OnWindowEnum, 0);
return windowList;
bool OnWindowEnum(int hwnd, int lparam)
{
// 可自行加入一些过滤条件。
windowList.Add(hwnd);
return true;
}
}
遍历具有指定类名或者标题的窗口
我们需要添加一些可以用于过滤窗口的 Win32 API。以下是我们即将用到的两个:
1
2
3
4
5
6
7
// 获取窗口的类名。
[DllImport("user32.dll")]
private static extern int GetClassName(int hWnd, StringBuilder lpString, int nMaxCount);
// 获取窗口的标题。
[DllImport("user32")]
public static extern int GetWindowText(int hwnd, StringBuilder lptrString, int nMaxCount);
于是根据类名找到窗口的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public static IReadOnlyList<int> FindWindowByClassName(string className)
{
var windowList = new List<int>();
EnumWindows(OnWindowEnum, 0);
return windowList;
bool OnWindowEnum(int hwnd, int lparam)
{
var lpString = new StringBuilder(512);
GetClassName(hwnd, lpString, lpString.Capacity);
if (lpString.ToString().Equals(className, StringComparison.InvariantCultureIgnoreCase))
{
windowList.Add(hwnd);
}
return true;
}
}
使用此方法,我们可以传入 "txguifoundation"
找到 QQ/TIM 的窗口:
1
var qqHwnd = FindWindowByClassName("txguifoundation");
要获取窗口的标题,或者把标题作为过滤条件,则使用 GetWindowText
。
在 QQ/TIM 中,窗口的标题是聊天对方的名字或者群聊名称。
1
2
var lptrString = new StringBuilder(512);
GetWindowText(hwnd, lptrString, lptrString.Capacity);
参考资料
本文会经常更新,请阅读原文: https://blog.walterlv.com/post/find-specific-window-by-enum-windows.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 吕毅 (包含链接: https://blog.walterlv.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 ([email protected]) 。