LastError:= GetLastError;
if LastError = 0 then Text:= 'Приложение не отвечает'
else
Text:= 'Ошибка при получении длины заголовка: ' + IntToStr(LastError);
end
else
begin
SetLength(Text, TextLen);
if TextLen > 0 then
if SendMessageTimeout(Wnd, WM_GETTEXT, TextLen + 1, LParam(Text),
SMTO_NORMAL or SMTO_ABORTIFHUNG, 5000, TextLen) = 0 then
begin
LastError:= GetLastError;
if LastError = 0 then Text:= 'Приложение не отвечает'
else Text:= 'Ошибка при получении заголовка:' + IntToStr(LastError);
end;
end;
Для каждого окна программа выводит имя оконного класса и реальный класс окна. В большинстве случаев эти два класса совпадают. Они различаются только для тех окон, чьи классы "унаследованы" от стандартных классов, таких как EDIT
, COMBOBOX
и т. п.
Вообще, наследования оконных классов в Windows нет. Но существует один нехитрый прием, который позволяет имитировать наследование. Оконная процедура обычно не обрабатывает все сообщения, а передает часть их в одну из стандартных оконных процедур (DefWindowProc
, DefFrameProc
и т. п.). Программа может с помощью функции GetClassInfo
узнать адрес оконной процедуры, назначенной стандартному классу, и использовать ее вместо стандартной оконной процедуры. Так как большая часть свойств окна определяется тем, как и какие сообщения оно обрабатывает, использование оконной процедуры другого класса позволяет почти полностью унаследовать свойства этого класса. (В VCL для наследования оконных классов существует метод TWinControl.CreateSubClass
.) Функция RealGetWindowClass
позволяет узнать имя класса-предка, если такой имеется. Соответствующая часть кода примера приведена в листинге 1.45.
GetClassName(Wnd, ClassName, ClassNameLen);
ClassName[ClassNameLen — 1]:= #0;
ListParams.Items[2].SubItems[0]:= ClassName;
RealGetWindowClass(Wnd, ClassName, ClassNameLen);
ClassName[ClassNameLen — 1]:= #0;
ListParams.Items[3].SubItems[0]:= ClassName;
У окна, если оно имеет стиль WS_CHILD
, должно быть родительское окно. Если такого стиля нет, то окно располагается непосредственно на рабочем столе. Кроме того, такое окно может (но не обязано) иметь владельца. Получить дескриптор родительского окна можно с помощью функции GetParent
. Владельца — с помощью функции GetWindow
с параметром GW_OWNER
.
Кроме GetParent
существует функция GetAncestor
, которая также возвращает дескриптор родительского окна, если она вызвана с параметром GA_PARENT
. Разница между этими функциями заключается в том. что для окон верхнего уровня (т. е. расположенных непосредственно на рабочем столе) GetParent
возвращает 0, a GetAncestor
— дескриптор рабочего стопа (этот дескриптор можно получить через функцию GetDesktopWindow
).
Значительную часть кода программы составляет анализ того, какие флаги присутствуют в стиле окна. В этом нет ничего сложного, но он громоздкий из-за большого числа флагов. Следует также учесть, что для стандартных классов одни и те же числовые значения могут иметь разный смысл. Так, например, константы ES_NOHIDESEL
и BS_LEFT
имеют одинаковые значения. Поэтому при расшифровке стиля следует также учитывать класс окна. Приводить здесь этот код мы не будем по причине его тривиальности. Его можно посмотреть в примере на компакт-диске.
1.3.2. Обобщающий пример 2 — Ассоциированные файлы и предотвращение запуска второй копии приложения