1、在创建的窗口类里面

需要继承CWindowWnd、INotifyUI

对于CWindowWnd里面的方法:

实现CWindowWnd的方法virtualLPCTSTRGetWindowClassName()const=0;,重载virtualUINTGetClassStyle()const;返回窗口的风格类型,重载virtualvoidOnFinalMessage(HWNDhWnd);,

对于INotifyUI里面的,只有一个唯一的抽象方法:

virtualvoidNotify(TNotifyUI&msg)=0;

当你实现了这个Notify,才能正常的接收、处理消息

来看看主要的消息处理部分:

void Notify(TNotifyUI& msg)
{
if( msg.sType == _T("windowinit") ) OnPrepare();
else if( msg.sType == _T("click") ) {
if( msg.pSender->GetName() == _T("insertimagebtn") ) {
CRichEditUI* pRich = static_cast<CRichEditUI*>(m_pm.FindControl(_T("testrichedit")));
if( pRich ) {
pRich->RemoveAll();
}
}
else if( msg.pSender->GetName() == _T("changeskinbtn") ) {
if( CPaintManagerUI::GetResourcePath() == CPaintManagerUI::GetInstancePath() )
CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath() + _T("skin\\FlashRes"));
else
CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath());
CPaintManagerUI::ReloadSkin();
}
}
}

分开来看:

a、

void Notify(TNotifyUI& msg)
{
if( msg.sType == _T("windowinit") ) OnPrepare();
}

窗口刚创建的时候,发出来的消息时windowsinit,所以这里进行OnPrepare();

void OnPrepare()
{
CSliderUI* pSilder = static_cast<CSliderUI*>(m_pm.FindControl(_T("alpha_controlor")));
if( pSilder ) pSilder->OnNotify += MakeDelegate(this, &CFrameWindowWnd::OnAlphaChanged);
pSilder = static_cast<CSliderUI*>(m_pm.FindControl(_T("h_controlor")));
if( pSilder ) pSilder->OnNotify += MakeDelegate(this, &CFrameWindowWnd::OnHChanged);
pSilder = static_cast<CSliderUI*>(m_pm.FindControl(_T("s_controlor")));
if( pSilder ) pSilder->OnNotify += MakeDelegate(this, &CFrameWindowWnd::OnSChanged);
pSilder = static_cast<CSliderUI*>(m_pm.FindControl(_T("l_controlor")));
if( pSilder ) pSilder->OnNotify += MakeDelegate(this, &CFrameWindowWnd::OnLChanged);
}

这种方法很简单,只是找到每一个CSliderUI,而且是通过CPaintManagerUI的FindControl方法(参数为xml中描述的每一个CSliderUI的名称)

找到之后进行:

if( pSilder ) pSilder->OnNotify += MakeDelegate(this, &CFrameWindowWnd::OnAlphaChanged);

主要是给这个CSliderUI加上消息的映射,MakeDelegate的两个参数分别是哪个UI控件(object)和需要绑定的方法

b、第二种事件类型

void Notify(TNotifyUI& msg)
{
else if( msg.sType == _T("click") ) {
else if( msg.pSender->GetName() == _T("changeskinbtn") ) {
if( CPaintManagerUI::GetResourcePath() == CPaintManagerUI::GetInstancePath() )
CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath() + _T("skin\\FlashRes"));
else
CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath());
CPaintManagerUI::ReloadSkin();
}
}
}

这里有这么多的事件要处理,就拿一个换肤的例子来看看(如上面的例子):

其实也很简单,这么多的代码,只是找到对应的资源路径(比如最上边的一个if语句,判断当前的资源路径是否和当前的程序实例的路径相等,如果相等就……),比如这里"skin\\FlashRes",前面再加上项目的路径,来构成一个绝对路径。

CPaintManagerUI::ReloadSkin();是用来,替换资源的,比如xml中有一个资源的是这样的:

我们的资源路径下面有:

那么执行了这个方面,在使用到winbk.bmp的地方就会被直接替换掉了。

2、那么我的小伙伴们都要奇怪了,这个程序在什么地方响应windows的消息呢?

看看这段代码:

	LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
if( uMsg == WM_CREATE ) {
m_pm.Init(m_hWnd);
CDialogBuilder builder;
CControlUI* pRoot = builder.Create(_T("test1.xml"), (UINT)0, NULL, &m_pm);
ASSERT(pRoot && "Failed to parse XML");
m_pm.AttachDialog(pRoot);
m_pm.AddNotifier(this);
Init();
return 0;
}
else if( uMsg == WM_DESTROY ) {
::PostQuitMessage(0L);
}
else if( uMsg == WM_ERASEBKGND ) {
return 1;
}
LRESULT lRes = 0;
if( m_pm.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;
return CWindowWnd::HandleMessage(uMsg, wParam, lParam);
}

一看这里的WM_CREATE,就知道他是对windows的消息进行响应的部分。

这里只看WM_CREATE消息,其他的部分要么是退出的消息,要么就是北京擦除的消息、让windows自动处理默认消息的。

而WM_CREATE消息

if( uMsg == WM_CREATE ) {
m_pm.Init(m_hWnd);
CDialogBuilder builder;
CControlUI* pRoot = builder.Create(_T("test1.xml"), (UINT)0, NULL, &m_pm);
ASSERT(pRoot && "Failed to parse XML");
m_pm.AttachDialog(pRoot);
m_pm.AddNotifier(this);
Init();
return 0;
}

可以看到首先需要对CPaintManagerUI进行初始化,窗口的句柄是在CWindowWnd中的

得到的,CFrameWindowWnd继承了CWindowsWnd,所以这个成员变量的值自然也就存在了。CDialogBuilder是一个对话框的创建者,他负责创建一个对话框,在以这个库做UI的窗体,都是以对话框的形式显示出来的。

CControlUI* pRoot = builder.Create(_T("test1.xml"), (UINT)0, NULL, &m_pm);

通过xml文件和CPaintManagerUI来创建一个对话框,创建出来的对话框作为顶层(其他的控件作为子层)保存在pRoot中。最后把这个对话框放入到渲染管理(CPaintManagerUI大管家)中,这样duilib就能自己去渲染出来了。m_pm.AddNotifier(this);对这个大管家设置消息响应的类。这样一个检查的初始化就完成了。

3、main函数

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPSTR /*lpCmdLine*/, int nCmdShow)
{
HRESULT Hr = ::CoInitialize(NULL);
CFrameWindowWnd* pFrame = new CFrameWindowWnd(); pFrame->Create(NULL, _T("这是一个最简单的测试用exe,修改test1.xml就可以看到效果"), UI_WNDSTYLE_FRAME, WS_EX_WINDOWEDGE);
pFrame->CenterWindow();
pFrame->ShowWindow(true);
CPaintManagerUI::MessageLoop(); ::CoUninitialize();
return 0;
}

CFrameWindowWnd*是我们重载的类,通过creat可以创建出来一个实例,CenterWindow()可以创建一个窗体,ShowWindow(true)可以把窗体给显示出来。CPaintManagerUI::MessageLoop()进行整个UI的消息循环。

CoInitialize和CoUninitialize是初始化com的。

最新文章

  1. cmder添加右键菜单
  2. 线段树(codevs1082)
  3. python 内建类型
  4. 技巧题---Single boy
  5. 无需操作系统和虚拟机,直接运行Python代码
  6. (转)Redis 集群方案
  7. Vim简明教程【CoolShell】
  8. ecshop的小总结
  9. POJ 3340 &amp;amp; HDU 2410 Barbara Bennett&amp;#39;s Wild Numbers(数学)
  10. 在MyEclipse 2014中给Spket增加ExtJS提示
  11. Linux发行版 CentOS6.5 禁用防火墙步骤
  12. MySql查询不区分大小写解决方案(两种)
  13. Luogu4587 [FJOI2016]神秘数
  14. Python 元组tuple相关知识
  15. vue 图片加载失败调用
  16. python 闯关之路二(模块的应用)
  17. CRT&amp;EXCRT 中国剩余定理及其扩展
  18. zabbix 添加被监控主机
  19. 集合框架-Map集合
  20. mui框架如何实现页面间传值

热门文章

  1. 【Python初级】StringIO和BytesIO读写操作的小思考
  2. TCP/IP——IP网络协议简记
  3. 通过因特网连接Beaglebone Black
  4. FastReport.Net使用:[26]数字格式
  5. 最大流KK算法
  6. vijos Warcraft III 守望者的烦恼
  7. 1 Spring4 之环境搭建和HelloWorld
  8. 线性DP POJ 1159 Palindrome
  9. php 导出excel
  10. Codeforces Round #302 (Div. 2) C. Writing Code 简单dp