一、保存this指针的预留位置
SetWindowLongPtr可以保存两个用户指针,GWLP_USERDATA和DWLP_USER字段。
GWLP_USERDATA是对用户透明的,用户可以随意更改,一般用于外部数据,不建议用作this指针。
DWLP_USER是私有的,可以用来保存this指针,对话框(DialogBoxParam和CreateDialogParam)可以直接用,如果用CreateWindowEx创建的窗口,需要在RegisterClass注册自己的窗口类时,把WNDCLASS::cbWndExtra设为DLGWINDOWEXTRA。
为方便起见,统一将WNDCLASS::cbWndExtra设为DLGWINDOWEXTRA,并将this指针用SetWindowLongPtr保存到DWLP_USER。
二、传递this指针的方法
分CreateWindowEx和DialogBoxParam/CreateDialogParam两种情况。
CreateWindowEx比较复杂,最后一个参数可传递一个用户指针,WndProc在接受到WM_CREATE后,这个指针保存在((LPCREATESTRUCT)lParam)->lpCreateParams中。
DialogBoxParam和CreateDialogParam比较简单,最后一个参数可传递一个用户指针,DlgProc接受到WM_INITDIALOG之后,指针直接保存在lParam中。
获取指针之后,使用SetWindowLongPtr(hwnd, DWLP_USER, (LONG_PTR)thisptr)将指针保存到DWLP_USER字段。
典型代码:
if (message == WM_CREATE)
SetWindowLongPtr(hwnd, DWLP_USER, (LONG_PTR)((LPCREATESTRUCT)lParam)->lpCreateParams);
if (message == WM_INITDIALOG)
SetWindowLongPtr(hwnd, DWLP_USER, (LONG_PTR)lParam);
三、将消息分发到MyWindowClass::WndProc的方式
这个比较简单,这里只讲普通窗口,对话框同理。
每次WndProc调用时(在WM_CREATE或WM_INITDIALOG的特殊处理代码之后)都用GetWindowLongPtr获取对象指针,然后转换为MyWindowClass *类型,最后直接调用它的成员函数即可。典型代码如下:
MyWindowClass *obj = (MyWindowClass *)GetWindowLongPtr(hwnd, DWLP_USER);
if (obj != NULL) return obj->WndProc(hwnd, message, wParam, lParam);
这样以来每次WndProc被调用,MyWindowClass::WndProc都会被调用,并且这个函数有正确的this指针。