超类化和子类化没有具体的代码,其实是一种编程技巧,在MFC和WTL中可以有不同的实现方法。
窗口子类化:
原理就是改变一个已创建窗口类的窗口过程函数。通过截获已创建窗口的消息,从而实现监视或修改已创建窗口类的行为属性。可以用来改变或者扩展一个已存在的窗口的行为,而不用重新开发。比如要获得那些预定义控件窗口类(按钮控件、编辑控件、列表控件、下 拉列表控件、静态控件和滚动条控件)的功能而又要修改它们的某些行为。
子类化的优点主要体现在以下两个方面:首先,它不需要创建新的窗口类,不需要了解一个窗口的窗口过程。这在原来的窗口函数是由别人编写,而且创建过程不可见的情况下非常有用;其次,子类化比较容易实现,因为所有要做的工作仅仅就是写一个窗口函数。
主要步骤为
- 截取该消息,阻止其向原窗口函数发送。
- 修改该消息。
- 修改完毕以后再向原窗口函数发送。
// 保存窗口默认的消息响应函数指针 WNDPROC pSubclassOldEditProc; // 用于替换子类化窗口的消息响应函数 LRESULT CALLBACK JcEditProcSubClass(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_CHAR: { ::MessageBox(hWnd, "WM_CHAR响应", "子类化", MB_OK); return 0; } //使用完后,消息发回原窗体 default: return ::CallWindowProc(pSubclassOldEditProc, hWnd, message, wParam, lParam); } } // 对创建好的窗体进行子类化代码 { // 创建 HWND hEdit = CreateWindowEx(NULL, "EDIT", "SubClass", WS_CHILD|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL, 100,120, 128, 16, hWnd, NULL, hInstance, NULL); //修改窗口属性,改变消息响应函数 pSubclassOldEditProc = (WNDPROC)::SetWindowLong(hEdit, GWL_WNDPROC, (DWORD)JcEditProcSubClass); // 显示 ShowWindow(hEdit, nCmdShow); UpdateWindow(hWnd); }
窗口超类化:
窗口超类化是在窗口类——WNDCLASS或WNDCLASSEX(非MFC类概念)级别进行的改变窗口类特征的。改变已有窗口类的行为属性。
- 通过调用 GetClassInfoEx 来获得想要进行超类化操作的窗口类的信息。函数GetClassInfoEx 需要一个指向 WNDCLASSEX 结构的指针,用于当成功返回时填入窗口类的信息。
- 按需要修改 WNDCLASSEX 结构的成员,其中有两个成员必须修改:
hInstance 存放程序的实例句柄
lpszClassName 指向一个新类名的指针
不必修改成员 lpfnWndProc,但大多数情况下还是需要的。但要记住如果要使用函数 CallWindowProc 调用老窗口的过程,那就必须保存成员 lpfnWndProc 的原值。 - 注册修改完的 WNDCLASSEX 结构,得到一个具有旧窗口类某些特性的新窗口类。
- 用新窗口类创建窗。
WNDPROC pSuperOldEditProc;// 保存窗口默认消息处理函数 // 用于替换的超类化消息响应函数 LRESULT CALLBACK JcEditProcSuper(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_CHAR: { ::MessageBox(hWnd, "WM_CHAR响应", "超类化", MB_OK); return 0; } default: return ::CallWindowProc(pSuperOldEditProc, hWnd, message, wParam, lParam); } } // 创建超类化控件代码 { // 取得原控件信息 WNDCLASSEX myeditClass; ::GetClassInfoEx(hInstance, "EDIT", &myeditClass); // 保存原控件默认消息处理函数 pSuperOldEditProc = myeditClass.lpfnWndProc; // 设置替换的消息处理函数 myeditClass.lpfnWndProc = JcEditProcSuper; // 指定新的窗口类名字 myeditClass.lpszClassName = "JcilyEdit"; // 设置结构体大小 myeditClass.cbSize = sizeof(WNDCLASSEX); // 注册新信息 RegisterClassEx(&myeditClass); // 创建 HWND hEdit = CreateWindowEx(NULL, myeditClass.lpszClassName, "SuperClass", WS_CHILD|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL, 100,100, 128, 16, hWnd, NULL, hInstance, NULL); // 显示 ShowWindow(hEdit, nCmdShow); UpdateWindow(hWnd); }
窗口子类化和超类化的区别
(1) 子类化修改窗口过程函数, 超类化修改窗口类(新的窗口类名)
(2) 子类化是在窗口实例级别上的,超类化是在窗口类(WNDCLASS)级别上的。
(3) 超类化可以完成比子类化更复杂的功能,在SDK范畴上,可以认为子类化是超类化的子集。
(4) 子类化只能改变窗口创建后的性质,对于窗口创建期间无能为力(无法截获ON_CREATE 事件),而超类化可以实现;超类化不能用于Windows创建的窗口,子类化可以。
(5) 超类化可以修改包含窗体背景等属性,而子类化不能。