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

2016-01-29 12:23 52 1 收藏

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

【 tulaoshi.com - C语言心得技巧 】

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


作者:luzhl

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com)

下载源代码


    一直以来,编写非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代码块。    charm_hook[40];protected:    //m_pfnWndProc指针指向Hook代码块的始地址。    //注册窗口类(WNDCLASSEX),或者子类化控件窗口,或者DialogBox显示对话框    //等需要窗口回调过程参数时,使用m_pfnWndProc作为参数。    WNDPROCm_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.      

来源:https://www.tulaoshi.com/n/20160129/1485858.html

延伸阅读
1. 传指针时,我们可以通过指针来修改它在外部所指向的内容。但假如要修改外部指针所指向的对象是不可能的。例如传递外部指针到函数内来分配空间,必须传递指针的指针或指针的引用。 2. char carry[10] = {0}; 编译器会将其后所有的东西都置0; !-- frame contents -- !-- /frame contents -- 3. 函数返回值为const时,返回...
一.什么是窗口类   在Windows中运行的程序,大多数都有一个或几个可以看得见的窗口,而在这些窗口被创建起来之前,操作系统怎么知道该怎样创建该窗口,以及用户操作该窗口的各种消息交给谁处理呢?所以VC在调用Windows的API(CreateWindow或者CreateWindowEx)创建窗口之前,要求程序员必须定义一个窗口类(不是传统C++意义上的类)来规...
我谈到让一个类支持隐式类型转换通常是一个不好的主意。当然,这条规则有一些例外,最普通的一种就是在创建数值类型时。例如,假如你设计一个用来表现有理数的类,答应从整数到有理数的隐式转换看上去并非不合理。 !-- frame contents -- !-- /frame contents -- 这的确不比 C++ 的内建类型从 int 到 double 的转换更不合理(...
    我们现在再来看看AfxDeferRegisterClass是什么样子的:   #define AfxDeferRegisterClass(fClass)   ((afxRegisteredClasses & fClass) ? TRUE:AfxEndDeferRegisterClass(fClass)     #define afxRegisteredClasses AfxGetModuleState()-m_fRegisteredClasses   BOOL AFXAPI AfxEndDefe...
在学习这一章内容前我们已经学习过了类的构造函数和析构函数的相关知识,对于普通类型的对象来说,他们之间的复制是很简单的,例如: int a = 10; int b =a; !-- frame contents -- !-- /frame contents -- 自己定义的类的对象同样是对象,谁也不能阻止我们用以下的方式进行复制,例如:#include iostream&n...