MFC - Mensajes y eventos

Una aplicación se compone de varios objetos. La mayoría de las veces, más de una aplicación se está ejecutando en la computadora y el sistema operativo se le pide constantemente que realice algunas asignaciones. Debido a que puede haber tantas solicitudes presentadas de manera impredecible, el sistema operativo deja que los objetos especifiquen lo que quieren, cuándo lo quieren y qué comportamiento o resultado esperan.

Visión general

  • El sistema operativo Microsoft Windows no puede predecir qué tipo de solicitudes debería atender un objeto y qué tipo de asignación necesitaría otro objeto.

  • Para gestionar todas estas asignaciones y solicitudes, los objetos envían mensajes.

  • Cada objeto tiene la responsabilidad de decidir qué mensaje enviar y cuándo.

  • Para enviar un mensaje, un control debe crear un evento.

  • Para hacer una distinción entre los dos, el nombre de un mensaje generalmente comienza con WM_, que significa Mensaje de ventana.

  • El nombre de un evento generalmente comienza con On, lo que indica una acción.

  • El evento es la acción de enviar el mensaje.

Mapa de mensajes

Dado que Windows es un sistema operativo orientado a mensajes, una gran parte de la programación para el entorno de Windows implica el manejo de mensajes. Cada vez que ocurre un evento, como una pulsación de tecla o un clic del mouse, se envía un mensaje a la aplicación, que luego debe manejar el evento.

  • Para que el compilador administre los mensajes, deben incluirse en la definición de clase.

  • los DECLARE_MESSAGE_MAP La macro debe proporcionarse al final de la definición de clase como se muestra en el siguiente código.

class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      DECLARE_MESSAGE_MAP()
};
  • Los mensajes reales deben aparecer justo encima de la línea DECLARE_MESSAGE_MAP.

  • Para implementar los mensajes, debe crear una tabla de mensajes que esté usando su programa.

  • Esta tabla utiliza dos macros delimitadores;

  • Comienza con un BEGIN_MESSAGE_MAP y termina con un END_MESSAGE_MAP macros.

  • La macro BEGIN_MESSAGE_MAP toma dos argumentos, el nombre de su clase y la clase MFC de la que derivó su clase como se muestra en el código siguiente.

#include <afxwin.h>
class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      DECLARE_MESSAGE_MAP()
};
CMainFrame::CMainFrame() {

   // Create the window's frame
   Create(NULL, L"MFC Messages Demo", WS_OVERLAPPEDWINDOW,
                                      CRect(120, 100, 700, 480), NULL);
}
class CMessagesApp : public CWinApp {
   public:
      BOOL InitInstance();
};
BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
END_MESSAGE_MAP()
BOOL CMessagesApp::InitInstance(){
   m_pMainWnd = new CMainFrame;
   m_pMainWnd->ShowWindow(SW_SHOW);
   m_pMainWnd->UpdateWindow();
   return TRUE;
}
CMessagesApp theApp;

Veamos un ejemplo simple creando un nuevo proyecto Win32.

Step 1 - Para crear un proyecto MFC, haga clic con el botón derecho en el proyecto y seleccione Propiedades.

Step 2 - En la sección izquierda, haga clic en Propiedades de configuración → General.

Step 3 - Seleccione la opción 'Usar MFC en DLL compartida' en la sección Valores predeterminados del proyecto y haga clic en Aceptar.

Step 4 - Necesitamos agregar un nuevo archivo fuente.

Step 5 - Haga clic con el botón derecho en su proyecto y seleccione Agregar → Nuevo elemento.

Step 6 - En la sección Plantillas, haga clic en Archivo C ++ (.cpp).

Step 7 - Haga clic en Agregar para continuar.

Step 8 - Ahora, agregue el siguiente código en el archivo * .cpp.

#include <afxwin.h>
class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      DECLARE_MESSAGE_MAP()
};

CMainFrame::CMainFrame() {
   // Create the window's frame
   Create(NULL, L"MFC Messages Demo", WS_OVERLAPPEDWINDOW,
      CRect(120, 100, 700, 480), NULL);
}

class CMessagesApp : public CWinApp {
   public:
      BOOL InitInstance();
};

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
END_MESSAGE_MAP()
BOOL CMessagesApp::InitInstance() {
   m_pMainWnd = new CMainFrame;
   m_pMainWnd->ShowWindow(SW_SHOW);
   m_pMainWnd->UpdateWindow();
   return TRUE;
}
CMessagesApp theApp;

Mensajes de Windows

Hay diferentes tipos de mensajes de Windows, como crear una ventana, mostrar una ventana, etc. Estos son algunos de los mensajes de Windows más utilizados.

Mensaje Entrada de mapa Descripción
WM_ACTIVATE ON_WM_ACTIVATE () El marco llama a esta función miembro cuando se activa o desactiva un objeto CWnd.
WM_ACTIVATEA PP ON_WM_ACTIVATEAPP () El marco llama a esta función miembro a todas las ventanas de nivel superior de la tarea que se está activando y a todas las ventanas de nivel superior de la tarea que se está desactivando.
WM_APPCOMM Y ON_WM_APPCOMMAND () El marco llama a esta función miembro cuando el usuario genera un evento de comando de aplicación.
WM_CANCELMODE WM_CANCELMODE () El marco llama a esta función miembro para informar a CWnd que cancele cualquier modo interno.
WM_CHILDACTIVATE ON_WM_CHILDACTIVATE () Si el objeto CWnd es una ventana secundaria de interfaz de múltiples documentos (MDI), el marco llama a OnChildActivate cuando el usuario hace clic en la barra de título de la ventana o cuando la ventana se activa, mueve o cambia de tamaño.
WM_CLIPBOAR DUPDATE ON_WM_CLIPBOARDUPDATE () El marco llama a esta función miembro cuando el contenido del portapapeles ha cambiado.
WM_CLOSE ON_WM_CLOSE () El marco llama a esta función miembro como una señal de que CWnd o una aplicación debe terminar.
WM_CONTEXTMENU ON_WM_CONTEXTMENU () Llamado por el marco cuando el usuario ha hecho clic con el botón derecho del mouse (clic derecho) en la ventana.
WM_COPYDATA ON_WM_COPYDATA () Esta función miembro es llamada por el marco para copiar datos de una aplicación a otra.
WM_CREATE ON_WM_CREATE () El marco llama a esta función miembro cuando una aplicación solicita que se cree la ventana de Windows llamando a la función miembro Create o CreateEx.
WM_CTLCOLOR ON_WM_CTLCOLOR () El marco llama a esta función miembro cuando está a punto de dibujarse un control secundario.
WM_DELETEITEM ON_WM_DELETEITEM () El marco llama a esta función miembro para informar al propietario de un cuadro de lista o cuadro combinado dibujado por el propietario que el cuadro de lista o cuadro combinado se ha destruido o que se han eliminado elementos.
WM_DESTROY ON_WM_DESTROY () El framework llama a esta función miembro para informar al objeto CWnd que se está destruyendo.
WM_DRAWITEM ON_WM_DRAWITEM () El marco llama a esta función miembro para el propietario de un control de botón dibujado por el propietario, control de cuadro combinado, control de cuadro de lista o menú cuando ha cambiado un aspecto visual del control o menú.
WM_DROPFILES ON_WM_DROPFILES () El marco llama a esta función miembro cuando el usuario suelta el botón izquierdo del mouse sobre una ventana que se ha registrado como destinataria de archivos caídos.
WM_ENABLE ON_WM_ENABLE () El marco llama a esta función miembro cuando una aplicación cambia el estado habilitado del objeto CWnd. Sintaxis.
WM_HELPINFO ON_WM_HELPINFO () Maneja la Ayuda F1 dentro de la aplicación (usando el contexto actual).
WM_HOTKEY ON_WM_HOTKEY () El marco llama a esta función miembro cuando el usuario presiona una tecla de acceso rápido en todo el sistema.
WM_HSCROLL ON_WM_HSCROLL () El marco llama a esta función miembro cuando el usuario hace clic en la barra de desplazamiento horizontal de una ventana.
WM_KEYDOWN ON_WM_KEYDOWN () El marco llama a esta función miembro cuando se presiona una tecla que no es del sistema.
WM_KEYUP ON_WM_KEYUP () El marco llama a esta función miembro cuando se libera una clave que no es del sistema.
WM_KILLFOCUS ON_WM_KILLFOCUS () El marco llama a esta función miembro inmediatamente antes de perder el foco de entrada.
WM_LBUTTONDBLCLK ON_WM_LBUTTONDBLCLK () El marco llama a esta función miembro cuando el usuario hace doble clic con el botón izquierdo del mouse.
WM_LBUTTONDOWN ON_WM_LBUTTONDOWN () El marco llama a esta función miembro cuando el usuario presiona el botón izquierdo del mouse.
WM_LBUTTONUP ON_WM_LBUTTONUP () El marco llama a esta función miembro cuando el usuario suelta el botón izquierdo del mouse.
WM_MBUTTONDBLCLK ON_WM_MBUTTONDBLCLK () El marco llama a esta función miembro cuando el usuario hace doble clic con el botón central del mouse.
WM_MBUTTONDOWN ON_WM_MBUTTONDOWN () El marco llama a esta función miembro cuando el usuario presiona el botón central del mouse.
WM_MBUTTONUP ON_WM_MBUTTONUP () El marco llama a esta función miembro cuando el usuario suelta el botón central del mouse.
WM_MENUSELECT ON_WM_MENUSELECT () Si el objeto CWnd está asociado con un menú, el marco llama a OnMenuSelect cuando el usuario selecciona un elemento del menú.
WM_MOUSEACTIVATE ON_WM_MOUSEACTIVATE () El marco llama a esta función miembro cuando el cursor está en una ventana inactiva y el usuario presiona un botón del mouse.
WM_MOUSEHOVER ON_WM_MOUSEHOVER () El marco llama a esta función miembro cuando el cursor se desplaza sobre el área del cliente de la ventana durante el período de tiempo especificado en una llamada anterior a TrackMouseEvent.
WM_MOUSEHWHEEL ON_WM_MOUSEHWHEEL () El marco llama a este miembro cuando la ventana actual está compuesta por Desktop Window Manager (DWM), y esa ventana está maximizada.
WM_MOUSELEAVE ON_WM_MOUSELEAVE () El marco llama a esta función miembro cuando el cursor abandona el área del cliente de la ventana especificada en una llamada anterior a TrackMouseEvent.
WM_MOUSEMOVE ON_WM_MOUSEMOVE () El marco llama a esta función miembro cuando se mueve el cursor del mouse.
WM_MOVE ON_WM_MOVE () El marco llama a esta función miembro después de que se haya movido el objeto CWnd.
WM_PAINT ON_WM_PAINT () El marco llama a esta función miembro cuando Windows o una aplicación solicita volver a pintar una parte de la ventana de una aplicación.
WM_SETFOCUS () ON_WM_SETFOCUS () El marco llama a esta función miembro después de obtener el foco de entrada.
WM_SIZE () ON_WM_SIZE () El marco llama a esta función miembro después de que el tamaño de la ventana haya cambiado.
WM_TIMER ON_WM_TIMER () El marco llama a esta función miembro después de cada intervalo especificado en la función miembro SetTimer utilizada para instalar un temporizador.
WM_VSCROLL ON_WM_VSCROLL () El marco llama a esta función miembro cuando el usuario hace clic en la barra de desplazamiento vertical de la ventana.
WM_WINDOWPOSCHANGED ON_WM_WINDOWPOSCHANGED () El marco llama a esta función miembro cuando el tamaño, la posición o el orden Z ha cambiado como resultado de una llamada a la función miembro SetWindowPos u otra función de administración de ventanas.

Veamos un ejemplo simple de creación de ventanas.

WM_CREATE - Cuando se crea un objeto, llamado ventana, el marco que crea los objetos envía un mensaje identificado como ON_WM_CREATE.

Step 1- Para crear ON_WM_CREATE, agregue afx_msg int OnCreate (LPCREATESTRUCT lpCreateStruct); antes de DECLARE_MESSAGE_MAP () como se muestra a continuación.

class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
      DECLARE_MESSAGE_MAP()
};

Step 2 - Agregue ON_WM_CREATE () después de BEGIN_MESSAGE_MAP (CMainFrame, CFrameWnd) y antes de END_MESSAGE_MAP ()

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
   ON_WM_CREATE()
END_MESSAGE_MAP()

Step 3 - Aquí está la implementación de OnCreate ()

int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) {
   // Call the base class to create the window
   if (CFrameWnd::OnCreate(lpCreateStruct) == 0) {

      // If the window was successfully created, let the user know
      MessageBox(L"The window has been created!!!");
      // Since the window was successfully created, return 0
      return 0;
   }
   // Otherwise, return -1
   return -1;
}

Step 4 - Ahora su archivo * .cpp se verá como se muestra en el siguiente código.

#include <afxwin.h>
class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
      DECLARE_MESSAGE_MAP()
};
CMainFrame::CMainFrame() {

   // Create the window's frame
   Create(NULL, L"MFC Messages Demo", WS_OVERLAPPEDWINDOW,
      CRect(120, 100, 700, 480), NULL);
}
class CMessagesApp : public CWinApp {
   public:
      BOOL InitInstance();
};
BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
   ON_WM_CREATE()
END_MESSAGE_MAP()
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) {
   // Call the base class to create the window
   if (CFrameWnd::OnCreate(lpCreateStruct) == 0) {
      // If the window was successfully created, let the user know
      MessageBox(L"The window has been created!!!");
      // Since the window was successfully created, return 0
      return 0;
   }
   // Otherwise, return -1
   return -1;
}
BOOL CMessagesApp::InitInstance() { 
   m_pMainWnd = new CMainFrame;
   m_pMainWnd -> ShowWindow(SW_SHOW);
   m_pMainWnd -> UpdateWindow();
   return TRUE;
}
CMessagesApp theApp;

Step 5 - Cuando se compile y ejecute el código anterior, verá el siguiente resultado.

Step 6 - Al hacer clic en Aceptar, se mostrará la ventana principal.

Mensajes de comando

Una de las principales características de una aplicación gráfica es presentar los controles y recursos de Windows que permiten al usuario interactuar con la máquina. Ejemplos de controles que aprenderemos son botones, cuadros de lista, cuadros combinados, etc.

Un tipo de recurso que presentamos en la lección anterior es el menú. Dichos controles y recursos pueden iniciar sus propios mensajes cuando el usuario hace clic en ellos. Un mensaje que emana de un control o recurso de Windows se denomina mensaje de comando.

Veamos un ejemplo simple de mensajes de comando.

Para proporcionar a su aplicación la capacidad de crear un nuevo documento, la clase CWinApp proporciona el método OnFileNew ().

afx_msg void OnFileNew();

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
   ON_COMMAND(ID_FILE_NEW, CMainFrame::OnFileNew)
END_MESSAGE_MAP()

Aquí está la definición del método:

void CMainFrame::OnFileNew() {
   // Create New file
}

Mensajes de teclado

UN keyboardes un objeto de hardware adjunto a la computadora. De forma predeterminada, se utiliza para ingresar símbolos, letras y otros caracteres reconocibles en un control. Cada tecla del teclado muestra un símbolo, una letra o una combinación de ellos, para dar una indicación de para qué se podría usar la tecla. El usuario normalmente presiona una tecla, que envía una señal a un programa.

Cada tecla tiene un código que el sistema operativo puede reconocer. Este código se conoce comovirtual key code.

No Señor. Constante / valor y descripción
1

VK_LBUTTON

Boton izquierdo del raton

2

VK_RBUTTON

Botón derecho del mouse

3

VK_CANCEL

Procesamiento de rotura de control

4

VK_MBUTTON

Botón central del mouse (mouse de tres botones)

5

VK_BACK

Tecla RETROCESO

6

VK_RETURN

Introducir clave

7

VK_TAB

Tecla de tabulación

8

VK_CLEAR

Tecla BORRAR

9

VK_SHIFT

Tecla Shift

10

VK_CONTROL

Tecla CTRL

11

VK_MENU

Tecla Alt

12

VK_PAUSE

Tecla PAUSA

13

VK_CAPITAL

Tecla CAPS LOCK

14

VK_ESCAPE

Tecla ESC

15

VK_SPACE

Barra espaciadora

dieciséis

VK_PRIOR

Tecla PAGE UP

17

VK_NEXT

Tecla PAGE DOWN

18

VK_END

Tecla FIN

19

VK_HOME

Llave de la casa

20

VK_LEFT

Tecla FLECHA IZQUIERDA

21

VK_UP

Tecla FLECHA ARRIBA

22

VK_RIGHT

Tecla de FLECHA DERECHA

23

VK_DOWN

Tecla FLECHA ABAJO

24

VK_SELECT

SELECCIONAR tecla

25

VK_PRINT

PRINT clave

26

VK_EXECUTE

Tecla EJECUTAR

27

VK_SNAPSHOT

Tecla PRINT SCREEN

28

VK_INSERT

Clave INS

29

VK_DELETE

Tecla DEL

30

VK_NUMPAD0

Tecla 0 del teclado numérico

31

VK_NUMPAD1

Tecla del teclado numérico 1

32

VK_NUMPAD2

Tecla del teclado numérico 2

33

VK_NUMPAD3

Tecla del teclado numérico 3

34

VK_NUMPAD4

Teclado numérico 4 teclas

35

VK_NUMPAD5

Teclado numérico 5 teclas

36

VK_NUMPAD6

Teclado numérico 6 tecla

37

VK_NUMPAD7

Teclado numérico 7 tecla

38

VK_NUMPAD8

Teclado numérico 8 teclas

39

VK_NUMPAD9

Tecla 9 del teclado numérico

40

VK_MULTIPLY

Multiplicar clave

41

VK_ADD

Agregar clave

42

VK_SEPARATOR

Llave separadora

43

VK_SUBTRACT

Restar clave

44

VK_DECIMAL

Clave decimal

45

VK_DIVIDE

Dividir clave

46

VK_F1

Tecla F1

47

VK_F2

Tecla F2

48

VK_F3

Tecla F3

49

VK_F4

Tecla F4

50

VK_F5

Tecla F5

52

VK_F6

Tecla F6

53

VK_F7

Tecla F7

54

VK_F8

Tecla F8

55

VK_F9

Tecla F9

56

VK_F10

Tecla F10

57

VK_F11

Tecla F11

58

VK_F12

Tecla F12

59

VK_NUMLOCK

Tecla NUM LOCK

60

VK_SCROLL

Tecla SCROLL LOCK

61

VK_LSHIFT

Tecla SHIFT izquierda

62

VK_RSHIFT

Tecla SHIFT derecha

63

VK_LCONTROL

Tecla CONTROL izquierda

64

VK_RCONTROL

Tecla CONTROL derecha

Al presionar una tecla, se coloca un mensaje WM_KEYDOWN o WM_SYSKEYDOWN en el mensaje del hilo. Esto se puede definir de la siguiente manera:

afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags);

Veamos un ejemplo sencillo.

Step 1 - Aquí está el mensaje.

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
   ON_WM_CREATE()
   ON_WM_KEYDOWN()
END_MESSAGE_MAP()

Step 2 - Aquí está la implementación de OnKeyDown ().

void CMainFrame::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags) {
   switch (nChar) {

      case VK_RETURN:
         MessageBox(L"You pressed Enter");
         break;
      case VK_F1:
         MessageBox(L"Help is not available at the moment");
         break;
      case VK_DELETE:
         MessageBox(L"Can't Delete This");
         break;
      default:
         MessageBox(L"Whatever");
   }
}

Step 3 - Cuando se compile y ejecute el código anterior, verá el siguiente resultado.

Step 4 - Cuando presione Entrar, mostrará el siguiente mensaje.

Mensajes del mouse

El mouse es otro objeto que se adjunta a la computadora que permite al usuario interactuar con la máquina.

  • Si se presionó el botón izquierdo del mouse, se envía un mensaje ON_WM_LBUTTONDOWN. La sintaxis de este mensaje es:

    • afx_msg void OnLButtonDown (UINT nFlags, CPoint point)

  • Si se presionó el botón derecho del mouse, se envía un mensaje ON_WM_RBUTTONDOWN. Su sintaxis es -

    • afx_msg void OnRButtonDown (UINT nFlags, CPoint point)

  • De manera similar, si se suelta el botón izquierdo del mouse, se envía el mensaje ON_WM_LBUTTONUP. Su sintaxis es -

    • afx_msg void OnLButtonUp (UINT nFlags, punto CPoint)

  • Si se suelta el botón derecho del ratón, se envía el mensaje ON_WM_TBUTTONUP. Su sintaxis es -

    • afx_msg void OnRButtonUp (UINT nFlags, punto CPoint)

Veamos un ejemplo sencillo.

Step 1 - Agregue las siguientes dos funciones en la definición de clase CMainFrame como se muestra en el siguiente código.

class CMainFrame : public CFrameWnd {
   public:
      CMainFrame();
   protected:
      afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags);
      afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
      afx_msg void OnRButtonUp(UINT nFlags, CPoint point);
      DECLARE_MESSAGE_MAP()
};

Step 2 - Agregue los siguientes dos mapas de mensajes.

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
   ON_WM_KEYDOWN()
   ON_WM_LBUTTONDOWN()
   ON_WM_RBUTTONUP()
END_MESSAGE_MAP()

Step 3 - Aquí está la definición de funciones.

void CMainFrame::OnLButtonDown(UINT nFlags, CPoint point) { 
   CString MsgCoord;
   MsgCoord.Format(L"Left Button at P(%d, %d)", point.x, point.y);
   MessageBox(MsgCoord);
}
void CMainFrame::OnRButtonUp(UINT nFlags, CPoint point) { 
   MessageBox(L"Right Mouse Button Up");
}

Step 4 - Cuando ejecute esta aplicación, verá el siguiente resultado.

Step 5 - Cuando haga clic en Aceptar, verá el siguiente mensaje.

Step 6- Haga clic derecho en esta ventana. Ahora, cuando suelte el botón derecho del mouse, mostrará el siguiente mensaje.