热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Windows–使用类的成员函数作为Win32窗口消息回调处理函数WindowProc

1 Win32窗口的创建过程学过Win32界面编程的都知道,创建一个Win32窗口一般经过以下几个步骤

1 Win32窗口的创建过程

学过Win32界面编程的都知道,创建一个Win32窗口一般经过以下几个步骤:



  • (1)定义窗口对象

  • (2)注册窗口

  • (3)创建窗口

  • (4)显示窗口

  • (5)消息处理函数

  • (6)消息循环


一个简单的Win32窗口创建代码与以下代码类似,







#include
//声明函数
LRESULT CALLBACK WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
//主函数WinMain()
int WINAPI WinMain(_In_ HINSTANCE hInstance, _In_opt_ HINSTANCE hPrevInstance,
_In_ LPSTR lpCmdLine, _In_ int nShowCmd)
{
//窗口类名
TCHAR cls_name[] = TEXT("MyWinClass");
//设计窗口类
WNDCLASSEX wce = { 0 };
wce.cbSize = sizeof(WNDCLASSEX); //结构体大小
wce.style = CS_HREDRAW | CS_VREDRAW; //大小改变,水平和垂直重绘
wce.lpfnWndProc = WindowProc; //窗口回调函数地址
wce.cbClsExtra = 0; //类的附加数据
wce.cbWndExtra = 0; //窗口的附加数据
wce.hInstance = hInstance; //应用程序实例句柄
wce.hIcon = LoadIcon(NULL, IDI_APPLICATION); //大图标
wce.hIcOnSm= wce.hIcon; //小图标
wce.hCursor = LoadCursor(NULL, IDC_ARROW); //光标
wce.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); //背景色
wce.lpszMenuName = NULL; //菜单
wce.lpszClassName = cls_name; //窗口类的名称
//注册窗口类
//ATOM nres = RegisterClassEx(&wce);
if (FALSE == RegisterClassEx(&wce))
return 1;
//创建窗口
HWND hWnd = CreateWindowEx(
WS_EX_APPWINDOW, //窗口的扩展样式
cls_name, //窗口类名
TEXT("我的窗口"), //窗口标题
WS_OVERLAPPEDWINDOW, //窗口风格样式
200, 120, 600, 400, //窗口x,y,宽度,高度
NULL, //父窗口句柄
NULL, //菜单句柄
hInstance, //应用程序实例句柄
NULL); //附加数据
//回调函数要写上默认的处理DefWindowProc,不然创建失败
if (!hWnd)
return 1;
//显示,更新窗口
ShowWindow(hWnd, SW_SHOW);
UpdateWindow(hWnd);
//添加加速键表
HACCEL hAccel=NULL;
/*::LoadAccelerators(hInstance,MAKEINTRESOURCE(IDR_ACCELERATOR1);*/
//消息循环
MSG msg;
while (GetMessage(&msg,NULL,0,0))
{
if(!TranslateAccelerator(hWnd,hAccel,&msg))//有加速键表时用这,没有就不用
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return 0;
}
//窗口消息处理函数
LRESULT CALLBACK WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_CLOSE:
DestroyWindow(hWnd);
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default: break;
}
return DefWindowProc(hWnd,uMsg,wParam,lParam);
}

C++


Copy


一般情况下我们都会使用一个静态函数作为窗口的消息处理函数,就如同上述代码所示。但是在某些特殊情况下我们需要将某个自定义窗口的消息处理函数不采用静态函数而采用某个类的成员函数,方便封装,这种情况下有没有什么骇客的方式可以实现呢?


2 使用类的成员函数作为Win32窗口消息处理函数WindowProc

2.1 实现方式


在本文,将介绍一种特殊的方式,该方式主要是使用了Delphi的MakeObjectInstance的思想。


工具WndProcRemapUtil.h的代码如下:


#ifndef WNDPROC_REMAP_UTIL_H_
#define WNDPROC_REMAP_UTIL_H_
#include
const SIZE_T PageSize = 4096;
//产生一个代理函数
template
static WNDPROC MakeObjectInstance(LPVOID AObject, T AMethod)
{
union
{
T MethodAddr;//成员函数指针
LPVOID NomralAddr;//正常指针
}ut;//因为VC不允许成员函数指针转换到普通指针。只能变通的通过union来实现
const unsigned char BlockCode[] = {
#ifdef _WIN64
0x55,//{ push rbp }
0x48, 0x83, 0xEC, 0x40,//{ sub rsp,0x40 }
0x48, 0x8B, 0xEC,//{ mov rbp,rsp }
0x48, 0x89, 0x4D, 0x50,//{ mov[rbp + 0x50],rcx }
0x89, 0x55, 0x58,//{ mov[rbp + 0x58],edx }
0x4C, 0x89, 0x45, 0x60,//{ mov[rbp + 0x60],r8 }
0x4C, 0x89, 0x4D, 0x68,//{ mov[rbp + 0x68],r9 }
0x48, 0xB9, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,//{ mov rcx,AObject }
0x48, 0x8B, 0x55, 0x50,//{ mov rdx,[rbp + 0x50] }
0x44, 0x8B, 0x45, 0x58,//{ mov r8,[rbp + 0x58] }
0x4C, 0x8B, 0x4D, 0x60,//{ mov r9,[rbp + 0x60] }
0x48, 0x8B, 0x45, 0x68,//{ mov rax,[rbp + 0x68] }
0x48, 0x89, 0x44, 0x24, 0x20,//{ mov[rsp + 0x20],rax }
0x49, 0xBB, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,//{ mov r11, AMethod }
0x49, 0xFF, 0xD3,//{ call r11 }
0x48, 0x8D, 0x65, 0x40,//{ lea rsp,[rbp + 0x40] }
0x5D,//{ pop rbp }
0xC3//{ ret }
#else
0x58,//{ pop eax }
0x68, 0x00, 0x00, 0x00, 0x00,//{ push AObject }
0x50,//{ push eax }
0xB8, 0x00, 0x00, 0x00, 0x00,//{ mov eax, AMethod }
0xFF, 0xE0//{ jmp eax }
#endif // endif
};
size_t CodeBytes = sizeof(BlockCode);
LPVOID Block = VirtualAlloc(nullptr, PageSize, MEM_COMMIT, PAGE_EXECUTE_READWRITE);
memcpy(Block, BlockCode, CodeBytes);
unsigned char* bBlock = (unsigned char*)Block;
ut.MethodAddr = AMethod;
#ifdef _WIN64
*PLONG64(&bBlock[25]) = LONG64(AObject);
*PLONG64(&bBlock[0x38]) = LONG64(ut.NomralAddr);
#else
*PLONG32(&bBlock[2]) = LONG32(AObject);
*PLONG32(&bBlock[8]) = LONG32(ut.NomralAddr);
#endif
return (WNDPROC)Block;
}
// 释放代理函数
static void FreeObjectInstance(WNDPROC wndProc)
{
VirtualFree(wndProc, PageSize, MEM_RELEASE);
}
#endif // !WNDPROC_REMAP_UTIL_H_

C++


Copy


2.2 使用方式


先在类A中声明一个成员函数作为窗口消息处理函数


class A
{
public:
A()
{
}
~A()
{
}
LRESULT CALLBACK OpenGLWndDisplayProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HDC hdc;
PAINTSTRUCT ps;
switch (message)
{
case WM_CREATE:
{
break;
}
case WM_SIZE:
{
break;
}
case WM_PAINT:
{
HDC hDC = ::GetDC(hWnd);
RECT rc = { 0 };
GetClientRect(hWnd, &rc);
HBRUSH hbr = ::CreateSolidBrush(RGB(0x00, 0xFF, 0x00)); // 绿色画刷
::FillRect(hDC, &rc, hbr);
::DeleteObject(hbr);
::ReleaseDC(hWnd, hDC);
break;
}
case WM_DESTROY:
{
break;
}
case WM_LBUTTONDOWN:
{
break;
}
case WM_LBUTTONUP:
{
break;
}
case WM_MOUSEMOVE:
{
break;
}
case WM_RBUTTONDOWN:
{
break;
}
case WM_RBUTTONUP:
{
break;
}
case WM_MBUTTONDOWN:
{
break;
}
case WM_MBUTTONUP:
{
break;
}
case WM_MOUSEWHEEL:
{
break;
}
case WM_KEYDOWN:
{
break;
}
default:
break;
}
return DefWindowProc(hWnd, message, wParam, lParam);
}
};

C++


Copy


然后在创建Win32窗口,注册窗口类时使用以下代码将A类的成员函数OpenGLWndDisplayProc作为窗口消息回调函数,


A a;
WNDPROC wndProc = MakeObjectInstance(&a, &A::OpenGLWndDisplayProc);
//设计窗口类
WNDCLASSEX wce = { 0 };
wce.cbSize = sizeof(WNDCLASSEX); //结构体大小
wce.style = CS_HREDRAW | CS_VREDRAW; //大小改变,水平和垂直重绘
wce.lpfnWndProc = wndProc; //窗口回调函数地址
wce.cbClsExtra = 0; //类的附加数据
wce.cbWndExtra = 0; //窗口的附加数据
wce.hInstance = hInstance; //应用程序实例句柄
wce.hIcon = LoadIcon(NULL, IDI_APPLICATION); //大图标
wce.hIcOnSm= wce.hIcon; //小图标
wce.hCursor = LoadCursor(NULL, IDC_ARROW); //光标
wce.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); //背景色
wce.lpszMenuName = NULL; //菜单
wce.lpszClassName = cls_name; //窗口类的名称
//注册窗口类
//ATOM nres = RegisterClassEx(&wce);
if (FALSE == RegisterClassEx(&wce))
return 1;

C++


Copy


最后在程序退出时释放


FreeObjectInstance(wndProc);


推荐阅读
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 本文介绍了指针的概念以及在函数调用时使用指针作为参数的情况。指针存放的是变量的地址,通过指针可以修改指针所指的变量的值。然而,如果想要修改指针的指向,就需要使用指针的引用。文章还通过一个简单的示例代码解释了指针的引用的使用方法,并思考了在修改指针的指向后,取指针的输出结果。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • Voicewo在线语音识别转换jQuery插件的特点和示例
    本文介绍了一款名为Voicewo的在线语音识别转换jQuery插件,该插件具有快速、架构、风格、扩展和兼容等特点,适合在互联网应用中使用。同时还提供了一个快速示例供开发人员参考。 ... [详细]
  • 展开全部下面的代码是创建一个立方体Thisexamplescreatesanddisplaysasimplebox.#Thefirstlineloadstheinit_disp ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
  • 本文介绍了UVALive6575题目Odd and Even Zeroes的解法,使用了数位dp和找规律的方法。阶乘的定义和性质被介绍,并给出了一些例子。其中,部分阶乘的尾零个数为奇数,部分为偶数。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • C++中的三角函数计算及其应用
    本文介绍了C++中的三角函数的计算方法和应用,包括计算余弦、正弦、正切值以及反三角函数求对应的弧度制角度的示例代码。代码中使用了C++的数学库和命名空间,通过赋值和输出语句实现了三角函数的计算和结果显示。通过学习本文,读者可以了解到C++中三角函数的基本用法和应用场景。 ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 本文介绍了win7系统休眠功能无法启动和关闭的解决方法,包括在控制面板中启用休眠功能、设置系统休眠的时间、通过命令行定时休眠、手动进入休眠状态等方法。 ... [详细]
  • 本文介绍了Composer依赖管理的重要性及使用方法。对于现代语言而言,包管理器是标配,而Composer作为PHP的包管理器,解决了PEAR的问题,并且使用简单,方便提交自己的包。文章还提到了使用Composer能够避免各种include的问题,避免命名空间冲突,并且能够方便地安装升级扩展包。 ... [详细]
  • 本文介绍了一种轻巧方便的工具——集算器,通过使用集算器可以将文本日志变成结构化数据,然后可以使用SQL式查询。集算器利用集算语言的优点,将日志内容结构化为数据表结构,SPL支持直接对结构化的文件进行SQL查询,不再需要安装配置第三方数据库软件。本文还详细介绍了具体的实施过程。 ... [详细]
author-avatar
笨雄不笨
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有