Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
WinAPI.docx
Скачиваний:
49
Добавлен:
16.12.2018
Размер:
3.43 Mб
Скачать

Пример программы

Программа, названная BEEPER1, устанавливает таймер на временной интервал, равный 1 секунде. При получении сообщения WM_TIMER программа изменяет цвет рабочей области с голубого на красный или с красного на голубой и, вызывая функцию MessageBeep, издает гудок.

В программе BEEPER1 таймер устанавливается в функции WinMain, а сообщения WM_TIMER обрабатываются в оконной процедуре WndProc. При обработке сообщения WM_TIMER программа BEEPER1 вызывает функцию MessageBeep, инвертирует значение bFlipFlop и, для генерации сообщения WM_PAINT, делает окно недействительным. При обработке сообщения WM_PAINT программа BEEPER1 с помощью вызова функции GetClientRect получает структуру RECT, соответствующую размерам окна целиком, и с помощью вызова функции FillRect, закрашивает окно.

Поскольку BEEPER1 издает звуковой сигнал при каждом получении сообщения WM_TIMER, вы сможете, загружая BEEPER1 и выполняя какие-то действия в Windows, получить представление о нерегулярной природе сообщений WM_TIMER. Например, попытайтесь сдвинуть или изменить размер окна программы BEEPER1. Это заставит программу войти в "модальный цикл обработки сообщений" (modal message loop). Windows предотвращает все действия, которые могут помешать операции перемещения или изменения размера окна, перехватывая все сообщения с помощью цикла обработки сообщений, организуемого в самой операционной системе, чтобы сообщения не поступали в цикл обработки сообщений вашей программы. Большинство сообщений, предназначенных для окна программы, проходя через этот цикл, просто отбрасываются, поэтому BEEPER1 перестает сигналить. Если вы прекратите двигать окно или менять его размер, то заметите, что BEEPER1 не получила всех пропущенных сообщений WM_TIMER, они были потеряны, хотя первые два сообщения могут появиться с интервалом менее одной секунды.

7.2.2 Второй способ

При первом способе установки таймера сообщения WM_TIMER посылаются в обычную оконную процедуру. С помощью второго способа вы можете заставить Windows пересылать сообщение таймера другой функции из вашей программы.

Функция, которая будет получать эти таймерные сообщения, называется функцией "обратного вызова" (call-back). Это функция вашей программы, которую вызывает Windows. Вы сообщаете Windows адрес этой функции, а позже Windows вызывает ее. Это должно быть вам знакомым, поскольку оконная процедура программы фактически является такой функцией обратного вызова. При регистрации класса окна вы сообщаете Windows адрес оконной процедуры, и затем Windows, посылая сообщения программе, вызывает эту функцию.

SetTimer — это не единственная функция Windows, использующая функцию обратного вызова. Функции CreateDialog и DialogBox (обсуждаемые далее) используют функции обратного вызова для обработки сообщений в окне диалога; несколько функций Windows (EnumChildWindows, EnumFonts, EnumObjects, EnumProps и EnumWindows) передают перечисляемую информацию функциям обратного вызова; для нескольких реже используемых функций (GrayString, LineDDA и SetWindowsHookEx) также требуются функции обратного вызова.

Также как и оконная процедура, функция обратного вызова должна определяться как CALLBACK, поскольку Windows вызывает ее вне кодового пространства программы. Параметры функции обратного вызова и ее возвращаемое значение зависят от назначения функции обратного вызова. В случае функции обратного вызова для таймера, входными параметрами являются те же параметры, что и параметры оконной процедуры. Таймерная функция обратного вызова не имеет возвращаемого в Windows значения.

Давайте назовем функцию обратного вызова TimerProc. (Вы можете дать ей любое имя.) Она будет обрабатывать только сообщения WM_TIMER.

VOID CALLBACK TimerProc(HWND hwnd, UINT iMsg, UINT iTimerID, DWORD dwTime)

{

[обработка сообщений WM_TIMER]

}

Входной параметр hwnd — это описатель окна, задаваемый при вызове функции SetTimer. Windows будет посылать функции TimerProc только сообщения WM_TIMER, следовательно, параметр iMsg всегда будет равен WM_TIMER. Значение iTimerID — это идентификатор таймера, а значение dwTime — системное время.

Как уже говорилось выше, для первого способа установки таймера требуется следующий вызов функции SetTimer:

SetTimer(hwnd, iTimerID, iMsecInterval, NULL);

При использовании функции обратного вызова для обработки сообщений WM_TIMER, четвертый параметр функции SetTimer заменяется адресом функции обратного вызова:

SetTimer(hwnd, iTimerID, iMsecInterval,(TIMERPROC) TimerProc);

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]