WTL for MFC Programming实践篇 --- 一个自定义ComboBox的移植过程(下)

发表于:2007-07-01来源:作者:点击数: 标签:
我们重新回到起点,来看看那里出了错。仔细地研读代码以后发现,事件是怎么传递到MSG_MAP的呢?难道我们通过赋值将一个窗体句柄传进来,我们在这个类中定义的MSG_MAP就能自动的连接到这个句柄上吗?这显然是真的不可能。 那么没有将MSG_MAP连接到窗体句柄很


    我们重新回到起点,来看看那里出了错。仔细地研读代码以后发现,事件是怎么传递到MSG_MAP的呢?难道我们通过赋值将一个窗体句柄传进来,我们在这个类中定义的MSG_MAP就能自动的连接到这个句柄上吗?这显然是真的不可能。

那么没有将MSG_MAP连接到窗体句柄很可能是控件类无法收到任何事件的原因。那么如何将MSG_MAP连接到窗体句柄上呢?原书中提到一个重要的函数,CWindowImpl::SubclassWindow()。我们再次更改我们的控件类:

CComboBoxEx& operator =(HWND hWnd) {

CWindowImpl< CComboBoxEx, CComboBox>::SubclassWindow(hWnd); 

return *this;

}

一测之下,大吃一惊。不仅重画事件被正确触发,连析构函数中的没有Attach的Detach这个怪用法也可以删除了。为什么会这样呢?探究这个问题之前,让我们先看看原书使用的DDX_CONTROL宏 - 它只针对CWindowImpl的派生类起作用 - 是怎么回事。原码如下:

#define DDX_CONTROL(nID, obj) \

if(nCtlID == (UINT)-1 || nCtlID == nID) \

DDX_Control(nID, obj, bSaveAndValidate);



// Full control subclassing (for CWindowImpl derived controls)

template <class TControl>

void DDX_Control(UINT nID, TControl& ctrl, BOOL bSave)

{

if(!bSave && ctrl.m_hWnd == NULL)

{

T* pT = static_cast<T*>(this);

ctrl.SubclassWindow(pT->GetDlgItem(nID));

}

}

从原码可以看到,DDX_CONTROL宏和DDX_CONTROL_HANDLER宏实现的区别只是,前者使用SubclassWindow,而后者使用操作符“=”。如果把我们上面的代码联系起来,在操作符“=”的处理函数中调用SubclassWindow,其实就等于是明着使用DDX_CONTROL_HANDLER宏,暗地里却把DDX_CONTROL宏实现了。原来想出门,结果先绕着后院跑了3圈,这真是一个大笑话。

为什么会这样呢?不使用DDX_CONTROL宏是因为CComboBox没有SubclassWindow函数,而是用CComboBox是因为在MFC中CComboBoxEx就是从CComboBox派生,移植的时候当然倾向于选择同名的类,而不是CWindowImpl<CComboBoxEx, CComboBox>这样怪怪的声明方法。

可是这里忽视了一个基本的WTL特性,由于WTL基于ATL,而设计ATL就是为了将接口和实现分开,所以在WTL中所有不带Impl字样的类都不是实现类,像CWindow,CButton,CComboBox等等,他们只是包含一个句柄,没有自己的事件,他们只是负责中转,封装控件事件等等。像CComboBox的操作符“=”就只是一个赋值语句而已。而DDX_CONTROL_HANDLER正是为这些类服务的,当然如果我们注意到这个宏得注释,也许早就发现这个问题了,还记得吗?在这里重温一下吧:

// Full control subclassing (for CWindowImpl derived controls)

template <class TControl>

void DDX_Control(UINT nID, TControl& ctrl, BOOL bSave)



// Simple control attaching (for HWND wrapper controls)

template <class TControl>

void DDX_Control_Handle(UINT nID, TControl& ctrl, BOOL bSave)

好了,在环游地球一周以后,我们又回到了起点,虽然费了不少的力气,但也搞清楚不少的东西,下面大概地总结一下:

  1.  WTL的类包含接口类(只包含窗体句柄和事件的封装)和实现类(可以拥有自己的事件),要根据具体情况有选择的使用。
  2.  WTL不会自动销毁窗体句柄(当然是指接口类),所以Attach操作以后要记着Detach
  3.  注意包含有HANDLE的宏,类,函数,它们往往是接口类或为接口类服务的,如上面所说的DDX_Control_Handle,以及CDCHandle等等。
  4.  DDX是通过宏定义重载CWinDataExchange::DoDataExchange()函数实现的
  5.  消息反射是在取道发送消息的窗体句柄后,通过像它回发相应的消息来实现的。
  6.  当你想说这不可能的时候,往往是你在调用的方法上出现了错误。
  7.  多看看代码,你会了解得更多

原文转自:http://www.ltesting.net