前言
最近为了能够共享windows应用窗口内容,需要遍历桌面所有窗口,其中也遇到了不少的坑,比如遇到了很多不需要的窗口,无标题窗口,windows设置,邮件和计算器。好在最后还是实现了。这里直接上主要功能代码,避免大家踩坑。
代码
windowIterator.h
class CWindowIterator
{
private:
static char* wchar2char(const wchar_t* wchar);
static BOOL CALLBACK EnumWindowsProc(HWND window, LPARAM lParam);
public:
using EnumCallback = std::function<void(std::string &str)>;
void EnumWindow(EnumCallback callback);
};
windowIterator.cpp
char* CWindowIterator::wchar2char(const wchar_t* wchar)
{
char* pchar;
int len = WideCharToMultiByte(CP_ACP, 0, wchar, wcslen(wchar), NULL, 0, NULL, NULL);
pchar = new char[len + 1];
WideCharToMultiByte(CP_ACP, 0, wchar, wcslen(wchar), pchar, len, NULL, NULL);
pchar[len] = '\0';
return pchar;
}
BOOL CALLBACK CWindowIterator::EnumWindowsProc(HWND window, LPARAM lParam)
{
if (!IsWindow(window))
return TRUE;
if (!IsWindowVisible(window))
return TRUE;
DWORD flag = 0;
DwmGetWindowAttribute(window, DWMWA_CLOAKED, &flag, sizeof(flag));
if (flag)
return TRUE;
UINT wnd_style_ex = GetWindowLong(window, GWL_EXSTYLE);
if ((WS_EX_TOOLWINDOW & wnd_style_ex) == WS_EX_TOOLWINDOW)
return TRUE;
TCHAR szTitle[MAX_PATH] = { 0 };
::GetWindowText(window, szTitle, MAX_PATH); // 获取标题
if (wcslen(szTitle) == 0)
return TRUE;
std::string str = wchar2char(szTitle);
if (lParam) {
EnumCallback* cb = reinterpret_cast<EnumCallback*>(lParam);
(*cb)(str);
}
return TRUE;
}
void CWindowIterator::EnumWindow(EnumCallback callback)
{
::EnumWindows(EnumWindowsProc, (LPARAM)&callback);
}
windowIteratorTest.cpp
#include <iostream>
#include "windowIterator.h"
int main()
{
CWindowIterator wi;
wi.EnumWindow([](std::string &str) {
std::cout << str << std::endl;
});
return 0;
}
总结
总结了一下过程,需要能够获取类似alt + tab一样的预览窗口,主要是注意以下几点
1) 窗口必须可见 - 使用 GetWindowVisible
2) 窗口不能是工具栏窗口 - 使用 GetWindowLong
3) 不得隐藏窗口 - 使用 DwmGetWindowAttribute