logo search
Конспект по ООПиП ч

19. Обработка сообщений. Карты сообщений.

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

Процедура окна с помощью библиотеки MFC создается автоматически. Если необходима собственная обработка сообщения, то создается функция обработки сообщения, являющаяся членом класса управления окном. Для определения обработчика сообщения можно воспользоваться ClassWizard, как описано ниже.

Например, если указатель находится внутри окна представления, то при нажатии левой кнопки мыши передается идентификатор WM_LBUTTONDOWN. Чтобы предусмотреть собственную обработку этого сообщения, используйте мастер ClassWizard для создания функции класса представления, обрабатывающей данное сообщение.

Командные сообщения. MFC обеспечивает специальную обработку сообщений, генерируемых объектами пользовательского интерфейса, стандартными элементами, поддерживаемыми библиотекой MFC: меню, комбинации клавиш, кнопки панелей инструментов, строки состояния, элементы управления диалоговых окон. Сообщения, генерируемые объектами пользовательского интерфейса, называют командными сообщениями. Каждый раз, когда пользователь выбирает объект интерфейса или когда один из этих объектов необходимо обновить, объект передает командное сообщение главному окну. Однако библиотека MFC сразу направляет сообщение объекту окна представления. Если он не имеет нужного обработчика, библиотека MFC направляет сообщение объекту документа. Если же объект документа не содержит обработчик, библиотека MFC направляет сообщение объекту главного окна программы. Если главное окно также не располагает обработчиком, сообщение направляется объекту приложения. Наконец, если объект приложения не обеспечивает обработку, то сообщение обрабатывается стандартным образом.

Таким образом, библиотека MFC расширяет основной механизм сообщений, чтобы командные сообщения обрабатывались не только объектами, управляющими ими, но и любыми другими объектами приложения. Каждый из них принадлежит классу, прямо или косвенно порожденному от класса CCmdTarget, реализующего механизм передачи сообщений.

Важной особенностью такого механизма является то, что программа может обрабатывать нужное сообщение внутри наиболее подходящего для этого класса. Например, в программе, созданной мастером AppWizard, команда Exit в меню File обрабатывается классом приложения, так как эта команда воздействует на приложение в целом. С другой стороны, команда Save в меню File обрабатывается классом Документа, так как этот класс отвечает за хранение и запись данных документа.

Схема сообщений. Когда мастер ClassWizard создает обработчик сообщения, то помимо объявления и определения функции-члена он также добавляет ее в специальную структуру MFC, называемую схемой (картой) сообщений (message map) и связывающую функции с обрабатываемыми сообщениями. Схема сообщений позволяет библиотеке MFC вызывать для каждого типа сообщения соответствующий обработчик.

Мастера AppWizard и ClassWizard создают необходимый код для реализации схемы сообщений, основанной на наборе MFC-макросов. После применения мастера ClassWizard для определения обработчиков трех видов сообщений мыши (создание которых рассматривается ниже) в файл CMiniDrawView.h будут добавлены следующие макросы и объявления функций:

// Сгенерированные функции схемы сообщений

protected:

//{{AFX_MSG(CMiniDrawView)

afx_msg void OnLButtonDown(UINT nFlags, CPoint point);

afx_msg void OnMouseMove(UINT nFlags, CPoint point);

afx_msg void OnLButtonOp(UINT nFlags, CPoint point);

//}}AFX_MSG DECLARE MESSAGE MAP()

В файл реализации представления MiniDrawView.cpp будут добавлены соответственно следующие макросы:

BEGIN_MESSAGE_MAP (CMiniDrawView, CView)

//{(AFX_MSG_MAP(CMiniDrawView)

ON_WM_LBUTTONDOWN ()

ON_WM_MOUSEMOVE ()

ON_WM_LBUTTONUP ()

//}}AFX_MSG_MAP END_MESSAGE_MAP()

Когда сообщение передается объекту класса, MFC обращается к схеме сообщений, чтобы определить, есть ли в классе обработчик такого сообщения. Если обработчик найден, ему передается управление. При отсутствии обработчика MFC ищет его в базовом классе. Если это не дает результата, то поиск будет продолжен по иерархии классов до первого встретившегося обработчика. Если в иерархии обработчик отсутствует, то будет выполнена стандартная обработка сообщения. Если же это командное сообщение, то оно перенаправляется следующему объекту в описанной ранее последовательности.

Обратите внимание: некоторые из классов библиотеки MFC, от которых порождены классы программы, содержат обработчики сообщений. Следовательно, даже если производный класс не определяет обработчик сообщения, обработчик базового класса может обеспечить соответствующую обработку. Например, базовый класс Документа CDocument содержит обработчики сообщений, поступающих при выборе команд Save, Save As... и Close в меню File (соответственно OnFileSave, OnFileSaveAs И OnFileClose).