当前位置: 首页 > 图文教程 > 开发语言 > VC++ > 使非MFC窗口程序的窗口回调过程成为C++类的成员函数

VC++
用 auto_ptr 类模板帮助动态内存管理
走近 STL
一步一步学STL标准模板库
使用 <multimap> 库创建重复键关联容器
使用 <map> 库创建关联容器
用 vectors 改进内存的再分配
用函数模板实现和优化抽象操作
STL 字符串类与 UNICODE 及其它......
如何在Dll中导出STL类
再谈“在STL列表(Lists)中插入不同类型的对象”
使用::std::vector<>作为管理动态数组的优先选择
三种常见中文内码的转换方法
JNI 中文处理问题小结
构建 GB2312 汉字库的 unicode 码表
正则表达式简介
在非MFC程序中引用CString
UTF-8与GB2312之间的互换
宽字符标量L"xx"在VC6.0/7.0和GNU g++中的不同实现
用VC++设计语法编辑器
C语言中对时间和日期的处理

VC++ 中的 使非MFC窗口程序的窗口回调过程成为C++类的成员函数


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-10-30   浏览: 101 ::
收藏到网摘: n/a

使非MFC窗口程序的窗口回调过程成为C++类的成员函数


作者:luzhl

下载源代码


    一直以来,编写非MFC下的窗口程序,都习惯把窗口过程及消息处理函数编写成全局函数。为了把窗口回调过程及窗口消息处理函数封装成C++窗口类的成员函数,于是我编写了抽象类CWndProc:

一、头文件

//wndpro.h#ifndef __WNDPROC_H__#define __WNDPROC_H__class CWndProc{protected: //保护的构造函数,必须由派生类来构造。 CWndProc(); virtual ~CWndProc();protected: //窗口回调过程,基类作为纯虚函数没有实现代码。 virtual LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) = 0;private: //Hook代码块。 char	m_hook[40];protected: //m_pfnWndProc指针指向Hook代码块的始地址。 //注册窗口类(WNDCLASSEX),或者子类化控件窗口,或者DialogBox显示对话框 //等需要窗口回调过程参数时,使用m_pfnWndProc作为参数。 WNDPROC	m_pfnWndProc;};#endif //__WNDPROC_H__//end of file 
二、实现代码文件
//wndproc.cpp#include "stdafx.h"#include "wndproc.h"/*全局的Hook代码,其C的伪代码为:LRSULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam){ return (CWndProc派生类的this指针)->WndProc(hwnd, uMsg, wParam, lParam);}代码的功能就是直接转调用CWndProc派生类的WndProc。*/static unsigned char g_hook[] ={ 0x8B, 0x44, 0x24, 0x10, // mov eax,dword ptr [esp+10h] ; eax <- lParam 0x8B, 0x4C, 0x24, 0x0C, // mov ecx,dword ptr [esp+0Ch] ; ecx <- wParam 0x8B, 0x54, 0x24, 0x08, // mov edx,dword ptr [esp+8] ; edx <- uMsg 0x50, // push eax ; lParam 参数入栈 0x8B, 0x44, 0x24, 0x08, // mov eax,dword ptr [esp+8] ; eax <- hwnd 0x51, // push ecx ; wParam 参数入栈 0xB9, 0x00, 0x00, 0x00, 0x00, // mov ecx,0 ; ecx <- this指针,这里暂时用this(NULL), // ; 在类构造函数初始化时修改为实际类的this指针值 0x52, // push edx ; uMsg 参数入栈 0x50, // push eax ; hwnd 参数入栈 0x51, // push ecx ; this 参数入栈 0xE8, 0x00, 0x00, 0x00, 0x00, // call WndProc ; 调用派生类的WndProc,这暂时用0, // ; 在类构造函数初始化时修改为实际类虚拟表WndProc指针偏移值 0xC2, 0x10, 0x00 // ret 10h ; return};CWndProc::CWndProc(){ char	*p; LRESULT (CALLBACK CWndProc::*pfn)(HWND, UINT, WPARAM, LPARAM); CopyMemory(m_hook, g_hook, sizeof(g_hook)); //把全局的Hook代码块,拷贝到类对象的Hook代码块 p = m_hook + 19; //p指针指向 mov ecx, 0 处,以便修改this(NULL)指针为实际类对象的this指针 *((unsigned int *)p) = (unsigned int)this; //修改p所指向的位置为mov ecx, (指向实际的类对象的this指针) pfn = WndProc; //pfn指向类虚拟表中WndProc函数的指针; p = m_hook + 27; //p指针指向 call WndProc处,以便修改WndProc虚表指针相对偏移值 //由于vc6.0无法修改pfn及强制其类型,所以下面使用几句汇编 __asm { mov eax, pfn	; eax <- pfn sub eax, 4 ; eax <- eax-4 mov edi, p ; edi <- p指针 sub eax, edi ; eax <- eax-edi 计算WndProc虚表指针与当前 EIP+5 相对偏移值 mov [edi], eax	; eax <- [edi] 修改p所指向的位置为 call (WndProc虚表指针与当前 EIP+5 相对偏移值) } m_pfnWndProc = (WNDPROC)&m_hook[0]; //把Hook代码块始址赋给m_pfnWndProc}CWndProc::~CWndProc(){}//enf of file.