Для перечисления всех окон верхнего уровня служит функция EnumWindows, однако, если мы просто вызовем эту функцию, то обнаружим, что она возвращает много больше окон, чем видно приложений на экране. Очевидно, мы должны игнорировать невидимые окна и окна, имеющие владельца (такие как диалоговые панели). Но даже и после этой фильтрации в списке будет несколько больше окон, чем отображает Task Manager.
Проведя несколько экспериментов с Windows NT Task Manager, удалось установить, что он игнорирует окна с пустым заголовком, и, как ни странно, окна с заголовком "Program Manager". Да-да, если вы создатите свое окно с таким заголовком, то оно не появится в списке приложений Task Manager. Обычно в системе есть только одно окно "Program Manager" – это то окно, на котором находится рабочий стол. Понятно, что пользователи не ассоциируют это окно с каким-либо приложением, для них оно является неотъемлемой частью компьютера, и поэтому Task Manager должен игнорировать это окно. Непонятно только, почему разработчики Task Manager решили определять это окно по его заголовку, а не по имени класса окна, которое есть "progman".
После того, как мы вставим все эти проверки, наш список уже ничем не будет отличаться от выводимого Task Manager. Ниже приведен код функции EnumApplications, которая реализует перечисление приложений. Интерфейс функции построен в стиле функций-перечислителей в Win32 API: она принимает указатель на пользовательскую функцию, которую вызывает для каждого перечисляемого приложения.
typedef BOOL (CALLBACK * PFNENUMAPP)(
IN HWND hWnd, // идентификатор главного окна приложения
IN LPCTSTR pszName, // название приложения
IN HICON hIcon, // иконка приложения
IN LPARAM lParam // пользовательский параметр
);
typedef struct _ENUMAPPDATA {
LPARAM lParam;
PFNENUMAPP pfnEnumApp;
} ENUMAPPDATA, * PENUMAPPDATA;
static BOOL CALLBACK EnumWindowsCallback(IN HWND hWnd, IN LPARAM lParam) {
PENUMAPPDATA pEnumData = (PENUMAPPDATA)lParam;
_ASSERTE(_CrtIsValidPointer(pEnumData, sizeof(ENUMAPPDATA), 1));
if (!IsWindowVisible(hWnd) || GetWindow(hWnd, GW_OWNER) != NULL) return TRUE;
TCHAR szClassName[80];
GetClassName(hWnd, szClassName, 80);
if (lstrcmpi(szClassName, _T("Progman")) == 0) return TRUE;
// получаем заголовок окна
TCHAR szText[256];
DWORD cchText = GetWindowText(hWnd, szText, 256);
if (cchText == 0) return TRUE;
HICON hIcon = NULL;
// получаем иконку окна
if (SendMessageTimeout(hWnd, WM_GETICON, ICON_SMALL, 0,
SMTO_ABORTIFHUNG|SMTO_BLOCK, 1000, (DWORD_PTR *)&hIcon)) {
if (hIcon == NULL) {
if (!SendMessageTimeout(hWnd, WM_GETICON, ICON_BIG, 0,
SMTO_ABORTIFHUNG|SMTO_BLOCK, 1000, (DWORD_PTR *)&hIcon)) hIcon = NULL;
}
} else hIcon = NULL;
if (hIcon == NULL) hIcon = (HICON)GetClassLong(hWnd, GCL_HICONSM);
if (hIcon == NULL) hIcon = (HICON)GetClassLong(hWnd, GCL_HICON);
if (hIcon == NULL) hIcon = LoadIcon(NULL, IDI_APPLICATION);
// вызываем пользовательскую функцию
return pEnumData->pfnEnumApp(hWnd, szText, hIcon, pEnumData->lParam);
}
BOOL EnumApplications(IN PFNENUMAPP pfnEnumApp, IN LPARAM lParam) {
_ASSERTE(pfnEnumApp!= NULL);
ENUMAPPDATA EnumData;
EnumData.pfnEnumApp = pfnEnumApp;
EnumData.lParam = lParam;
return EnumWindows(EnumWindowsCallback, (LPARAM)&EnumData);
}
Как видно, функция EnumApplicationsчрезвычайно проста – она просто вызывает EnumWindowsи вся основная работа по фильтрации ненужных окон ложится на вспомогательную функцию EnumWindowsCallback.
В функции EnumWindowsCallbackмы сначала отсеиваем невидимые окна, и окна, имеющие владельца. Затем мы проверяем, не является ли данное окно окном рабочего стола. Здесь мы не уподобляемся разработчикам Windows NT Task Manager и используем имя класса окна для проверки. Наконец, мы отбрасываем окна с пустым заголовком.
После того, как мы определи, что данное окно представляет некоторое приложение, мы собираем информацию об окне, чтобы передать ее пользовательской функции. Сначала мы получаем заголовок окна с помощью хорошо известной функции GetWindowText. Затем мы пытаемся получить иконку окна. Обратите внимание, мы используем функцию SendMessageTimeoutс флагом SMTO_ABORTIFHUNG для посылки сообщения WM_GETICON. Это гарантирует, что наше приложение не зависнет, даже если приложение, которому принадлежит окно, перестало обрабатывать сообщения.
Когда все параметры определены, мы вызываем пользовательскую функцию. Пользовательская функция, в свою очередь, может распоряжаться этими данными по своему усмотрению. Например, в тестовом приложении Process Viewer, которое сопровождает эту статью, она добавляет очередной элемент в список приложений.
Cсылки
1. Q175030 HOWTO: Enumerate Applications in Win32, Microsoft Knowledge Base.
Тема: ООП и наследование
Читать дальше