走进windows编程的世界-----消息处理函数(2)

一 WM_PAINT消息

1 WM_PAINT的产生
   由于窗口的互相覆盖等,产生需要绘制的区域,那么会产生WM_PAINT消息.
   一般情况下,不直接发送WM_PAINT消息,通过API声明需要绘制区域,来产生WM_PAINT消息.
   例如,可以使用InvalidateRect声明一个需要重新绘制的区域.
  
  2 WM_PAINT的注意点
    2.1 如果一个消息队列中,有多个WM_PAINT消息,只有最后一个WM_PAINT消息会被处理.
    2.2 WM_PAINT消息处理中,要清空需要被绘制的区域. BeginPaint
      
  3 WM_PAINT的使用
    3.1 WM_PAINT开始时,必须调用BeginPaint
    3.2 绘制图形
    3.3 WM_PAINT处理后,必须调用EndPaint

看下面的示例代码:

/* File : winpaint.cpp
 * Auth : sjin
 * Date : 20140706
 * Mail : [email protected]
 */

#include <Windows.h>
#include <stdio.h>

HINSTANCE g_hInst = NULL;

void OnPaint(HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam)
{
    PAINTSTRUCT ps = {0};
	/*WM_PAINT开始时,必须调用的*/
    HDC hDC = BeginPaint(hWnd, &ps);

    CHAR szText[] = "Hello WM_PAINT";
	/*在指定的坐标下输出一行字*/
    TextOut(hDC, 100, 100, szText, strlen(szText));

	/*绘制一个矩形*/
    Rectangle(hDC, 200, 200, 300, 300);

	EndPaint(hWnd,&ps);
}

LRESULT CALLBACK WndProc(HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam)
{
    switch(nMsg)
    {
    case WM_PAINT:
        OnPaint(hWnd, nMsg, wParam, lParam);
		break;
    case WM_DESTROY:
        PostQuitMessage(0);
		return 0;
    }
    return DefWindowProc(hWnd, nMsg, wParam, lParam);
}

BOOL RegisterWnd(LPSTR lpszClassName)
{
    WNDCLASSEX wce = {0};
    wce.cbClsExtra = 0;
    wce.cbSize = sizeof(wce);
    wce.cbWndExtra = 0;
    wce.hbrBackground = HBRUSH(COLOR_BTNFACE + 1);
    wce.hCursor = NULL;
    wce.hIcon = NULL;
    wce.hIconSm = NULL;
    wce.hInstance = g_hInst;
    wce.lpfnWndProc = WndProc;
    wce.lpszClassName = lpszClassName;
    wce.lpszMenuName = NULL;
    wce.style = CS_HREDRAW|CS_VREDRAW;

    ATOM nAtom = RegisterClassEx(&wce);
    if(0 == nAtom)
    {
        return FALSE;
    }
    return TRUE;
}

HWND CreateWnd(LPSTR lpszClassName)
{
    HWND hWnd = CreateWindowEx(0,
        lpszClassName, "MyWndPaint", WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
        NULL, NULL, g_hInst, NULL);
    return hWnd;
}

void DisplayWnd(HWND hWnd)
{
    ShowWindow(hWnd, SW_SHOW);
    UpdateWindow(hWnd);
}

void Message()
{
    MSG msg = {0};
    while(GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
}

int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
     g_hInst = hInstance;
    RegisterWnd("MyWndPaint");
    HWND hWnd = CreateWnd("MyWndPaint");
    DisplayWnd(hWnd);
    Message();
    return 0;
}

三 键盘消息

1 键盘消息
   按键消息
     WM_KEYDOWN 当键被按下时产生
     WM_KEYUP 当键被释放时产生
     WM_SYSKEYDOWN 当系统键被按下时产生 ALT/F10
     WM_SYSKEYUP 当系统键释放时产生
    字符消息
     WM_CHAR 当有字符键被按下时产生
      TranslateMessage会将WM_KEYDOWN消息中,可以显示的按键,转换成WM_CHAR的消息
  2 消息参数
    WPARAM - 虚拟键码
    LPARAM - 相关的按键信息.
  
  3 消息的使用
    3.1 当有按键消息时,首先进入系统消息队列,然后被程序的消息循环获取.
    3.2 消息的处理
  4 键盘消息的顺序
    对于可显示字符: WM_KEYDOWN,WM_CHAR,WM_KEYUP
    对于不可显示字符: WM_KEYDOWN,WM_KEYUP
    对于系统键:WM_SYSKEYDOWN,WM_SYSKEYUP
    如果按键一直不释放,会重复产生
         WM_KEYDOWN(WM_CHAR)

/* File : winKeyboard.cpp
 * Auth : sjin
 * Date : 2014-07-06
 * Mail : [email protected]
 */

#include <Windows.h>
#include <stdio.h>

HINSTANCE g_hInst = NULL;
HANDLE g_hStdOut = NULL;

LRESULT CALLBACK WinProc(HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam)
{
    switch(nMsg)
    {
    case WM_KEYDOWN:/*键盘按下*/
        {
            CHAR szText[] = "WM_KEYDOWN\n";
            switch(wParam)
            {
            case VK_LEFT:
            case VK_RIGHT:
            case VK_UP:
            case VK_DOWN:
                WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL);
                break;
            }
        }
        break;
    case WM_KEYUP:/*键盘抬起*/
        {
            CHAR szText[] = "WM_KEYUP\n";
            WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL); break;
        }
        break;
    case WM_SYSKEYDOWN:/*系统键按下*/
        {
            CHAR szText[] = "WM_SYSKEYDOWN\n";
            WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL); break;
        }
        break;
    case WM_SYSKEYUP:/*系统键抬起*/
        {
            CHAR szText[] = "WM_SYSKEYUP\n";
            WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL); break;
        }
        break;
    case WM_CHAR:/*字符键盘*/
        {
            CHAR szText[260] = {0};
            sprintf(szText, "WM_CHAR: %c\n", wParam);
            WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL); break;
        }
        break;
    case WM_DESTROY:
        PostQuitMessage(0); return 0;
    }
    return DefWindowProc(hWnd, nMsg, wParam, lParam);
}

BOOL ResiterWnd(LPSTR lpszClassName)
{
    WNDCLASSEX wce = {0};
    wce.cbClsExtra = 0;
    wce.cbSize = sizeof(wce);
    wce.cbWndExtra = 0;
    wce.hbrBackground = HBRUSH(COLOR_BTNFACE+1);
    wce.hCursor = NULL;
    wce.hIcon = NULL;
    wce.hIconSm = NULL;
    wce.hInstance = g_hInst;
    wce.lpfnWndProc = WinProc;
    wce.lpszClassName = lpszClassName;
    wce.lpszMenuName = NULL;
    wce.style = CS_HREDRAW|CS_VREDRAW;

    ATOM nAtom = RegisterClassEx(&wce);
    if(0==nAtom)
    {
        return FALSE;
    }
    return TRUE;
}

HWND CreateWnd(LPSTR lpszClassName)
{
    HWND hWnd = CreateWindowEx(0, lpszClassName, "MyWnd",
        WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,CW_USEDEFAULT,
        NULL, NULL, g_hInst, NULL);
    return hWnd;
}

void ShowWnd(HWND hWnd)
{
    ShowWindow(hWnd, SW_SHOW);
    UpdateWindow(hWnd);
}

void Message()
{
    MSG msg = {0};
    while(GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
}

void NewConsole()
{
	/*产生控制台*/
    AllocConsole();

	/*获得控制台标准输出流句柄*/
    g_hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
    CHAR szText[] = "Debug Message......:\n";
	/*将szText 写到控制台*/
    WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL);
}

int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
	/*产生命令行窗口,用来调试程序*/
    NewConsole();
     g_hInst = hInstance;
    ResiterWnd("MyWnd");
    HWND hWnd = CreateWnd("MyWnd");
    ShowWnd(hWnd);
    Message();

    return 0;
}

三  鼠标消息

1 鼠标消息
    1) 基本鼠标消息
    WM_LBUTTONDOWN   左键按下
    WM_LBUTTONUP     左键抬起
    WM_RBUTTONDOWN   右键按下
    WM_RBUTTONUP     右键抬起
    WM_MOUSEMOVE     鼠标移动
    2) 双击消息
    WM_LBUTTONDBLCLK 左键双击 
    WM_RBUTTONDBLCLK 右键双击
    3) 滚轮消息
    WM_MOUSEWHEEL    鼠标滚轮

2 消息的参数
    WPARAM - 当前键盘和鼠标按键状态,例如MK_CONTROL/MK_SHIFT,MK_LBUTTON等
    LPARAM - 当前鼠标的坐标,坐标的原点是窗口客户区的左上角.
       X坐标 - LOWORD(lParam),低16位
       Y坐标 - HIWORD(lParam),高16位
    参数具体内容和具体鼠标消息有稍微不同. 
    
  3 消息的使用
  
    3.1 基本鼠标消息,只需在窗口处理函数增加消息处理即可. 当消息来临,获取鼠标和按键状态.例如:

case WM_MOUSEMOVE:
{
    int nX = LOWORD(lParam);
    int nY = HIWORD(lParam);
}

附:坐标转换的函数 ClientToScreen可以将鼠标坐标转换为屏幕的坐标.

3.2 双击消息
   3.2.1 窗口注册要增加 CS_DBLCLKS 类型
    wce.style = CS_DBLCLKS|...;
   3.2.2 在窗口处理函数中增加消息处理
   3.2.3 产生过程,例如:WM_LBUTTONDBLCLK
      WM_LBUTTONDOWN
      WM_LBUTTONUP
      WM_LBUTTONDBLCLK
      WM_LBUTTONUP
      连续两次LBUTTONDOWN的时间间隔小于预定的双击时间间隔,就会产生LBUTTONDBLCLK消息. 
      双击时间间隔可以通过控制面板调整.

3.3 滚轮消息
  3.3.1 由于WM_MOUSEWHEEL需要Winnt4.0以上版本支持,所以需要包含在windows.h的头文件前,增加 _WIN32_WINNT 宏定义,
        #define _WIN32_WINNT 0x0400
    3.3.2 在窗口处理函数中增加消息处理
    3.3.3 参数
     LPARAM 与其它鼠标消息类同
     WPARAM - LOWORD(WPARAM) 表示按键状态
              HIWORD(WPARAM) 滚轮滚动幅度,
                120的倍数,可以为正负值.
        正值: 滚轮向上滚动, 一般窗口向上滚动
        负值: 滚轮向下滚动, 一般窗口向下滚动

/* File : winCursor.cpp
 * Auth : sjin
 * Date : 2014-07-06
 * Mail : [email protected]
 */

#include <Windows.h>
#include <stdio.h>

int g_nXPos = 0;
int g_nYPos = 0;
int g_nX1Rect = 0;
int g_nY1Rect = 0;
int g_nX2Rect = 0;
int g_nY2Rect = 0;

HINSTANCE g_hInst   = NULL;
HANDLE    g_hStdOut = NULL;

void PrintLog(LPSTR lpszLog)
{
    WriteConsole(g_hStdOut, lpszLog, strlen(lpszLog), NULL, NULL);
}

void OnPaint(HWND hWnd, UINT nMsg, WPARAM wParam, LPARAM lParam)
{
    PAINTSTRUCT ps = {0};
    HDC hDC = BeginPaint(hWnd, &ps);
    CHAR szText[] = "Hello Mouse 12123";
    TextOut(hDC, g_nXPos, g_nYPos, szText, strlen(szText));

    Rectangle(hDC, g_nX1Rect, g_nY1Rect, g_nX2Rect, g_nY2Rect);
    EndPaint(hWnd, &ps);
}

LRESULT CALLBACK WndProc( HWND hWnd,
                          UINT nMsg,
                          WPARAM wParam,
                          LPARAM lParam )
{
    switch( nMsg )
    {
    case WM_PAINT:
        OnPaint(hWnd, nMsg, wParam, lParam); break;
    case WM_LBUTTONDOWN:/*鼠标左键按下*/
        {
            PrintLog("WM_LBUTTONDOWN\n");
            g_nX1Rect = LOWORD(lParam);
            g_nY1Rect = HIWORD(lParam);
        }
        break;
    case WM_LBUTTONUP:/*鼠标左键抬起*/
        {
            PrintLog("WM_LBUTTONUP\n");
            g_nX2Rect = LOWORD(lParam);
            g_nY2Rect = HIWORD(lParam);
			/*窗口更新区域添加一个矩形,背景将会擦出
			 * 会触发WM_PAINT消息。
			 */
            InvalidateRect(hWnd, NULL, TRUE);
        }
        break;
    case WM_RBUTTONDOWN:/*鼠标右键按下*/
        PrintLog("WM_RBUTTONDOWN\n");
        break;
    case WM_RBUTTONUP:/*鼠标右键抬起*/
        PrintLog("WM_RBUTTONUP\n");
        break;
    case WM_MOUSEMOVE:/*鼠标移动*/
        {
            int nX = LOWORD(lParam);
            int nY = HIWORD(lParam);
            POINT ptScreen = {0};
            ptScreen.x = nX;
            ptScreen.y = nY;
			/*函数将指定点,或者矩形的用户坐标转换成屏幕坐标*/
            ClientToScreen(hWnd, &ptScreen);

            CHAR szText[260] = { 0 };
            sprintf( szText,
                "WM_MOUSEMOVE: X=%d(%d),Y=%d(%d)\n",
                nX, ptScreen.x, nY, ptScreen.y );
            PrintLog( szText );
            if( wParam & MK_CONTROL )
            {
                PrintLog( "WM_MOUSEMOVE: MK_CONTROL\n" );
            }
            if( wParam & MK_LBUTTON )
            {
                PrintLog( "WM_MOUSEMOVE: MK_LBUTTON\n" );
            }

            g_nXPos = LOWORD(lParam);
            g_nYPos = HIWORD(lParam);
            //InvalidateRect( hWnd, NULL, TRUE );
        }
        break;
    case WM_LBUTTONDBLCLK:/*左键双击*/
        PrintLog( "WM_LBUTTONDBLCLK\n" );
        break;
    case WM_RBUTTONDBLCLK:/*右键双击*/
        PrintLog( "WM_RBUTTONDBLCLK\n" );
        break;
    case WM_MOUSEWHEEL:/*鼠标滚轮移动*/
        {
            short nDelta = HIWORD(wParam);
            int nX = LOWORD(lParam);
            int nY = HIWORD(lParam);
            CHAR szText[260] = { 0 };
            sprintf( szText,
                "WM_MOUSEWHEEL: Detla=%d, X=%d,Y=%d\n",
                nDelta, nX, nY );
            PrintLog( szText );
        }
        break;
    case WM_DESTROY:
        PostQuitMessage( 0 );
        break;
    }
    return DefWindowProc( hWnd, nMsg,
        wParam, lParam );
}

BOOL RegisterWnd( LPSTR pszClassName )
{
    WNDCLASSEX wce = { 0 };
    wce.cbSize          = sizeof( wce );
    wce.cbClsExtra      = 0;
    wce.cbWndExtra    = 0;
    wce.hbrBackground = HBRUSH(COLOR_WINDOW);
    wce.hCursor          = NULL;
    wce.hIcon          = NULL;
    wce.hIconSm       = NULL;
    wce.hInstance      = g_hInst;
    wce.lpfnWndProc   = WndProc;
    wce.lpszClassName = pszClassName;
    wce.lpszMenuName  = NULL;
    wce.style         = CS_HREDRAW|CS_VREDRAW|CS_DBLCLKS;
    ATOM nAtom = RegisterClassEx( &wce );
    if( 0 == nAtom )
    {
        return FALSE;
    }
    return TRUE;
}

HWND CreateWnd( LPSTR pszClassName )
{
    HWND hWnd = CreateWindowEx( 0,
        pszClassName, "MyWnd",
        WS_OVERLAPPEDWINDOW, CW_USEDEFAULT,
        CW_USEDEFAULT, CW_USEDEFAULT,
        CW_USEDEFAULT, NULL, NULL,
        g_hInst, NULL );
    return hWnd;
}

void DisplayWnd( HWND hWnd )
{
    ShowWindow( hWnd, SW_SHOW );
    UpdateWindow( hWnd );
}

void Message( )
{
    MSG msg = { 0 };
    while( GetMessage( &msg, NULL, 0, 0 ) )
    {
        TranslateMessage( &msg );
        DispatchMessage( &msg );
    }
}

void NewConsole()
{
	/*产生控制台*/
    AllocConsole();

	/*获得控制台标准输出流句柄*/
    g_hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
    CHAR szText[] = "[sjin] Debug Message......:\n";
	/*将szText 写到控制台*/
    WriteConsole(g_hStdOut, szText, strlen(szText), NULL, NULL);
}

int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
     NewConsole( );
     g_hInst = hInstance;
    RegisterWnd( "MyWnd" );
    HWND hWnd = CreateWnd( "MyWnd" );
    DisplayWnd( hWnd );
    Message( );

    return 0;
}

走进windows编程的世界-----消息处理函数(2)

时间: 2024-10-12 17:02:53

走进windows编程的世界-----消息处理函数(2)的相关文章

走进windows编程的世界-----消息处理函数(1)

Win32消息机制 过程驱动:程序是按照我们预先定义好的顺序执行,每执行一步,下一步都已经按照预定的顺序 继续执行,直至程序结束. 事件驱动:程序的执行顺序是无序的.某个时间点所执行的代码,是由外界 通知.由于我们无法决定程序执行顺序.所以代码的执行也是无序的. Win32基本消息 WM_DESTROY:       窗口销毁时的消息,可以做退出或善后处理 WM_CREATE:       窗口创建消息,是在窗口创建后,窗口处理函数收到的第一条消息   可以在这个消息内,做初始化或者穿件子窗口

走进windows编程的世界-----消息处理函数(4)

一 右键菜单  1 右键菜单    当在窗口点击鼠标右键时,弹出的菜单.  2 右键菜单的使用    2.1 创建菜单      CreatePopupMenu    2.2 菜单增加     AppendMenu    2.3 菜单的显示. BOOL TrackPopupMenu( HMENU hMenu, //显示的菜单句柄 UINT uFlags, //显示的方式 int x, //菜单的X屏幕坐标 int y, //菜单的Y屏幕坐标 int nReserved, //保留,必须为0 HW

走进windows编程的世界-----消息处理函数(3)

二 定时器消息 1 定时器消息 WM_TIMER   按照定时器设置时间段,自动向窗口发送一个定时器消息WM_TIMER. 优先级比较低.   定时器精度比较低,毫秒级别.消息产生时间也精度比较低.    2 消息和函数   2.1 WM_TIMER  - 消息ID    wParam: 定时器的ID    lParam: 定时器的处理函数 2.2 SetTimer  - 设置一个定时器 UINT SetTimer( HWND hWnd, //窗口的句柄,可以为NULL UINT nIDEven

走进windows编程的世界-----窗口的注册及创建

1   窗口注册和创建 1.1WIN32 窗口程序创建步骤 1.WinMain入口函数的定义 2.WindowProc函数的定义 3.注册窗口类 RegisterClass.RegisterClassEX 4.创建窗口 CreateWindow.CreateWindowEx HWND CreateWindow( LPCTSTRlpClassName,//指向已注册的窗口类的名称的指针 LPCTSTRlpWindowName,//指向窗口名称的指针 DWORDdwStyle,//窗口的风格 int

走进windows编程的世界-----windows进程

Windows进程  1 Windows进程    进程是一个容器,包含了一个应用程序实例的各种资源.Windows多任务的操作系统,因此可以同时执行多个进程.      2 Windows进程的一些特点    2.1 进程中包含了执行代码等资源.    2.2 进程都具有私有的地址空间.    2.3 每个进程都有一个ID,标识进程.    2.4 每个进程都有自己的安全属性    2.5 至少要包含一个可以执行的线程.    二 进程的环境 1 环境信息的获取    获取:    LPVOI

走进windows编程的世界-----字符编码

1   字符编码 1.1编码的历史 1.1.1ASCII码 0=127 7位表示 1.1.2ASCII扩展码 0-255 8为表示. 代码页:通过代码也来切换对应的字符(数字表示) 1.1.3双字节字符集DBCS 使用一个或两个字节表示字符. 1.1.4Unicode编码 全部使用2个字节表示字符 内存 硬盘等资源占用变大.对编码支持度大. 字符集 1.2C 语言和编码 1.2.1单字节的字符和字符串 Char  cText = 'A'; Char * pszText ="ABCD"

走进windows编程的世界-----绘图相关

Windows绘图 1 图形绘制      1.1 图形绘制的方式      获取到绘图句柄-设备描述表(DC),使用相应的绘图的API,在设备上绘制图形.          1.2 颜色      R\G\B三色, 每种颜色8位, 共24位颜色.      32位颜色: 颜色数量24位颜色, 多出来的8位表示灰度.      16位: 颜色数量2的16次方.            Win32下,颜色的定义 COLORREF(DWORD), RGB宏定义颜色       COLORREF nCo

走进windows编程的世界-----位图及映射模式

1   位图的使用 1.1位图介绍 位图-通过保存在图像上每个点的颜色,生成响应的位图文件. 光栅图:图像点阵的保存 矢量图:绘图命令的保存. 1.2位图的使用 1.             加载位图资源 LoadBitap(); 2.             创建防止位图的DC CreateCompatibleDC 3.             将位图放入创建的DC SlectObject 4.             绘制位图到当前DC中 BitBlt 5.             取出位图

走进windows编程的世界-----入门篇

1   Windows编程基础 1.1Win32应用程序基本类型 1)  控制台程序 不须要完好的windows窗体,能够使用DOS窗体方式显示 2)  Win32窗体程序 包括窗体的程序,能够通过窗体与程序进行交互 3)  Win32库程序 提供已有的代码,供其它程序使用 动态库(DLL):是在运行的时候能够载入的. 静态库(LIB):是在编译链接是使用的程序.成为当前程序的一部分. 1.2头文件和库 1.2.1头文件 主要的头文件windows.h包括了windows经常使用的定义等,其它,