1、首先来看这里的CRichListWnd
已经不再是从CWindowWnd继承了
classCRichListWnd:publicWindowImplBase
从WindowImplBase中,可以看到有三个抽象函数:
virtualCDuiStringGetSkinFolder()=0;
virtualCDuiStringGetSkinFile()=0;
virtualLPCTSTRGetWindowClassName(void)const=0;
这些都很简单,只是返回固定的string
2、看看WindowImplBase
- class UILIB_API WindowImplBase
- : public CWindowWnd
- , public CNotifyPump
- , public INotifyUI
- , public IMessageFilterUI
- , public IDialogBuilderCallback
- {
- ……
- }
这个类其实也是实现了CWindowWnd、INotifyUI,即一个窗体和消息,所以他含有virtualvoidNotify(TNotifyUI&msg);
但是这里又继承了CNotifyPump,所以这里的Notify可以这么实现:
- void WindowImplBase::Notify(TNotifyUI& msg)
- {
- return CNotifyPump::NotifyPump(msg);
- }
NotifyPump主要是遍历所有的虚拟窗口把消息传递出去
下面的代码列出了CRichListWnd比较重要的函数:
- class CRichListWnd : public WindowImplBase
- {
- public:
-
- virtual void OnFinalMessage( HWND );
- virtual void Notify( TNotifyUI &msg );
- virtual LRESULT OnMouseWheel( UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled);
- virtual LRESULT OnSysCommand( UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled );
- virtual void InitWindow();
- virtual LRESULT OnMouseHover( UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled );
- virtual LRESULT OnChar( UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled );
-
- DUI_DECLARE_MESSAGE_MAP()
- virtual void OnClick(TNotifyUI& msg);
- virtual void OnSelectChanged( TNotifyUI &msg );
- virtual void OnItemClick( TNotifyUI &msg );
- };
因为WindowImplBase类实现了CWindowWnd中的HandleMessage,所以我们如果需要把一些重要的消息事件给重写就行;
可以看到WindowImplBase中的OnCreate():
- LRESULT WindowImplBase::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
- {
- ……
- InitWindow();
- return 0;
- }
所以我们需要把InitWindow给重写了
在InitWindow给每一个控件和XML的控件属性对应起来:
- void CRichListWnd::InitWindow()
- {
- m_pCloseBtn = static_cast<CButtonUI*>(m_PaintManager.FindControl(_T("closebtn")));
- m_pMaxBtn = static_cast<CButtonUI*>(m_PaintManager.FindControl(_T("maxbtn")));
- m_pRestoreBtn = static_cast<CButtonUI*>(m_PaintManager.FindControl(_T("restorebtn")));
- m_pMinBtn = static_cast<CButtonUI*>(m_PaintManager.FindControl(_T("minbtn")));
- }
这就完成了初始化,窗口也就起来了
这个Demo中,是使用到了虚拟页的,也就是CPage1,如果不使用,他的消息会回到CRichListWnd中被处理,但是如果CPage1中处理了消息,CRichListWnd就不做处理了
我们来看看CPage1所要处理的消息:
同样的CRichListWnd消息
当页面发生OnClick或者OnItemClick时候,已经挂上了虚拟页CPage1,那么CRichListWnd消息的就失效了
3、关于虚拟页CPage的
这样挂上虚拟页的:
- CRichListWnd::CRichListWnd(void)
- {
- m_Page1.SetPaintMagager(&m_PaintManager);
-
- AddVirtualWnd(_T("page1"),&m_Page1);
- }
- CRichListWnd::~CRichListWnd(void)
- {
- RemoveVirtualWnd(_T("page1"));
- }
看到这里有个宏(因为继承了CNotifyPump都可以这样用):
- DUI_BEGIN_MESSAGE_MAP(CPage1, CNotifyPump)
- DUI_ON_MSGTYPE(DUI_MSGTYPE_CLICK, OnClick)
- DUI_ON_MSGTYPE(DUI_MSGTYPE_SELECTCHANGED, OnSelectChanged)
- DUI_ON_MSGTYPE(DUI_MSGTYPE_ITEMCLICK, OnItemClick)
- DUI_END_MESSAGE_MAP()
这是整个消息映射的,表面我们关心这几个消息函数,同样的还可以在UIDefine.h中找到其他的消息类型的事件
在void CPage1::OnItemClick( TNotifyUI &msg )中原来是没有代码的,需要手动把CRichListWnd的那部分粘上去(或者去掉挂上虚拟页的部分)
之后我们点击进度条的时候,这个函数才会执行
同样的把鼠标停留 在操作按钮上面会执行函数CRichListWnd::OnMouseHover