一 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 : 413977243@qq.com
*/ #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 : 413977243@qq.com
*/ #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 : 413977243@qq.com
*/ #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)的更多相关文章

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

    Win32消息机制     过程驱动:程序是依照我们预先定义好的顺序运行.每运行一步,下一步都已经依照预定的顺序 继续运行,直至程序结束.     事件驱动:程序的运行顺序是无序的.某个时间点所运行的 ...

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

    二 定时器消息 1 定时器消息 WM_TIMER   依照定时器设置时间段,自己主动向窗体发送一个定时器消息WM_TIMER. 优先级比較低.   定时器精度比較低,毫秒级别.消息产生时间也精度比較低 ...

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

    1   Windows编程基础 1.1Win32应用程序基本类型 1)  控制台程序 不须要完好的windows窗体,能够使用DOS窗体方式显示 2)  Win32窗体程序 包括窗体的程序,能够通过窗 ...

  4. 走进windows编程的世界-----画图相关

    Windows画图 1 图形绘制      1.1 图形绘制的方式      获取到画图句柄-设备描写叙述表(DC),使用对应的画图的API,在设备上绘制图形.          1.2 颜色     ...

  5. 走进windows编程的世界-----对话框、文本框、button

    1 对话框的分类  2 对话框的基本使用方式  3 对话框资源  4 有模式对话框的使用 int DialogBox( HINSTANCE hInstance, LPCTSTR lpTemplate, ...

  6. 走进windows编程的世界-----窗体的注冊及创建

    1   窗体注冊和创建 1.1WIN32 窗体程序创建步骤 1.WinMain入口函数的定义 2.WindowProc函数的定义 3.注冊窗体类 RegisterClass.RegisterClass ...

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

    Windows进程  1 Windows进程    进程是一个容器,包括了一个应用程序实例的各种资源.Windows多任务的操作系统,因此能够同一时候运行多个进程.      2 Windows进程的 ...

  8. [C#] 走进异步编程的世界 - 在 GUI 中执行异步操作

    走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5877042.html 序 这是继<开始接 ...

  9. 走进异步编程的世界 - 在 GUI 中执行异步操作

    转载:https://www.cnblogs.com/liqingwen/p/5877042.html 走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://w ...

随机推荐

  1. 分库代价高的情况下,如何优化ES解决亿级数据量检索

    数据平台已迭代三个版本,从一开始遇到很多常见的难题,到现在终于有片段时间整理一些已完善的文档,在此分享以供所需朋友的实现参考,但愿能帮助大家少走些弯路,在此篇幅中偏重于ElasticSearch的优化 ...

  2. Microsoft Updateclient更新

     大家好, 微软Microsoft Update产品组官方博客于昨天宣布了有关最新的Windows Updateclient更新的消息.依据这则博客.微软从当日開始逐渐向全部Windows 7, ...

  3. POJ 2041 Unreliable Message

    简单模拟.依照题意处理一下字符串就可以. 应该是写题号写错了,本来我在VirtualJudge是加入的POJ 并查集与生成树的题. #include<cstdio> #include< ...

  4. linux下开发,解决cocos2d-x中编译出现的一个小问题, undefined reference to symbol &#39;pthread_create@@GLIBC_2.2.5&#39;

    解决cocos2d-x中编译出现的一个小问题 对于cocos2d-x 2.×中编译中,若头文件里引入了#include "cocos-ext.h",在进行C++编译的时候会遇到例如 ...

  5. Oracle Study之--Oracle 单实例11.2.0.1.0升级到11.2.0.3.0

    Oracle Study之--Oracle 单实例11.2.0.1.0升级到11.2.0.3.0 系统环境: 操作系统:RedHat EL6(64位) Oracle:    Oracle 11gR2 ...

  6. Perl怎样过滤html标签

    比方一串字符串 <div><b>123</b></div> 假设仅仅想拿到123怎么办呢? 用perl的正則表達式能够非常easy的做到. $str = ...

  7. 2016 ICPC CAMP Recording

    等了好久终于等到今天 马上能和群巨们一起学习了 希望不要暴露我太弱的本质............ 北京不冷,就是风大~~~ 1.24 8点准时起床了,准备下楼吃早饭 (这个宾馆好多美美的空姐对面就是东 ...

  8. iOS开发- 生成/解析.vcf文件

    vcf, 通讯录导出的一种格式. 一.生成vcf文件 假设要把我们iPhone通讯录里的数据, 生成vcf格式文件. 我们能够借助iCloud. 小技巧:通过iCloud导出iPhone通讯录的方法 ...

  9. JAVA配置Tomcat

    1.下载tomcat,我jdk是1.8的,网上查了一下,说要安装tomcat8及以上的tomcat 尝试点击,弹出, 2.配置环境 3.安装通过cmd安装 4.点击开启服务 5.输入localhost ...

  10. zzulioj--1719--小胖的疑惑(整数划分+dp打表)

    1719: 小胖的疑惑 Time Limit: 1 Sec  Memory Limit: 128 MB Submit: 108  Solved: 51 SubmitStatusWeb Board De ...