Duilib经典实例教程;1基本框架;一个简单的Duilib程序一般是下面这个样子的:;//Duilib使用设置部分;#pragmaonce;#defineWIN32_LEAN_AND_ME;#define_CRT_SECURE_NO_DE;#include<windows.h>;#include<objbase.h>;#include&


Duilib经典实例教程

1 基本框架

一个简单的Duilib程序一般是下面这个样子的:

// Duilib使用设置部分

#pragma once

#define WIN32_LEAN_AND_MEAN

#define _CRT_SECURE_NO_DEPRECATE

#include <windows.h>

#include <objbase.h>

#include "..\DuiLib\UIlib.h"

using namespace DuiLib;

#ifdef _DEBUG

# ifdef _UNICODE

# pragma comment(lib, "..\\bin\\DuiLib_ud.lib")

# else

# pragma comment(lib, "..\\bin\\DuiLib_d.lib")

# endif

#else

# ifdef _UNICODE

# pragma comment(lib, "..\\bin\\DuiLib_u.lib")

# else

# pragma comment(lib, "..\\bin\\DuiLib.lib")

# endif

#endif

// 窗口实例及消息响应部分

class CFrameWindowWnd : public CWindowWnd, public INotifyUI

{

public:

CFrameWindowWnd() { };

LPCTSTR GetWindowClassName() const { return _T("UIMainFrame"); };

UINT GetClassStyle() const { return UI_CLASSSTYLE_FRAME | CS_DBLCLKS; };

void OnFinalMessage(HWND /*hWnd*/) { delete this; };

void Notify(TNotifyUI& msg)

{

if( msg.sType == _T("click") ) {

if( msg.pSender->GetName() == _T("closebtn") ) {

Close();

}

}

}

LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)

{

if( uMsg == WM_CREATE ) {

m_pm.Init(m_hWnd);

CControlUI *pButton = new CButtonUI;

pButton->SetName(_T("closebtn"));

pButton->SetBkColor(0xFFFF0000);

m_pm.AttachDialog(pButton);

m_pm.AddNotifier(this);

return 0;

}

else if( uMsg == WM_DESTROY ) {

::PostQuitMessage(0);

}

LRESULT lRes = 0;

if( m_pm.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;

return CWindowWnd::HandleMessage(uMsg, wParam, lParam);

}

public:

CPaintManagerUI m_pm;

};

// 程序入口及Duilib初始化部分

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPSTR /*lpCmdLine*/, int nCmdShow)

{

CPaintManagerUI::SetInstance(hInstance);

CPaintManagerUI::SetResourcePath(CPaintManagerUI::GetInstancePath());

CFrameWindowWnd* pFrame = new CFrameWindowWnd();

if( pFrame == NULL ) return 0;

pFrame->Create(NULL, _T("测试"), UI_WNDSTYLE_FRAME, WS_EX_WINDOWEDGE); pFrame->ShowWindow(true);

CPaintManagerUI::MessageLoop();

return 0;

}

可以看出,这个程序分三个部分:

?

?

? Duilib使用设置部分,这个部分都是一些使用Duilib所需要的头文件和自动链接到相应的Duilib库,一般来说基本上不用改动。 窗口实例及消息响应部分,基本的窗口实现类和简单的消息响应,需要重点关注的是void Notify(TNotifyUI& msg)中的事件处理,这是Duilib程序最重要的部分。 程序入口及Duilib初始化部分,Duilib初始化和窗口创建。

编译这个程序,出现如下效果:

点击红色区域的任意位置,窗口会立即关闭。这样我们就完成了最简单的一个Duilib程序编写,虽然这个例子还不能展现Duilib的强大,但也算是麻雀虽小,肝胆俱全了。

因为很多美观的界面都不使用系统的标题栏和边框这些非客户区绘制,我们也把这掉,修改CFrameWindowWnd:: HandleMessage为:

LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)

{

if( uMsg == WM_CREATE ) {

m_pm.Init(m_hWnd);

CControlUI *pButton = new CButtonUI;

pButton->SetName(_T("closebtn"));

pButton->SetBkColor(0xFFFF0000);

m_pm.AttachDialog(pButton);

m_pm.AddNotifier(this);

return 0;

}

else if( uMsg == WM_DESTROY ) {

::PostQuitMessage(0);

}

else if( uMsg == WM_NCACTIVATE ) {

if( !::IsIconic(m_hWnd) ) {

return (wParam == 0) ? TRUE : FALSE;

}

}

else if( uMsg == WM_NCCALCSIZE ) {

return 0;

}

else if( uMsg == WM_NCPAINT ) {

return 0;

}

LRESULT lRes = 0;

if( m_pm.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;

return CWindowWnd::HandleMessage(uMsg, wParam, lParam);

}

编译运行,我们得到如下效果:

2 编写界面xml

为了使用xml进行界面布局,需要把前面的Duilib程序框架中的HandleMessage稍微改动一下:

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);

return 0;

}

else if( uMsg == WM_DESTROY ) {

::PostQuitMessage(0);

}

else if( uMsg == WM_NCACTIVATE ) {

if( !::IsIconic(m_hWnd) ) {

return (wParam == 0) ? TRUE : FALSE;

}

}

else if( uMsg == WM_NCCALCSIZE ) {

return 0;

}

else if( uMsg == WM_NCPAINT ) {

return 0;

}

LRESULT lRes = 0;

if( m_pm.MessageHandler(uMsg, wParam, lParam, lRes) ) return lRes;

return CWindowWnd::HandleMessage(uMsg, wParam, lParam);

}

灰色部分表示改动的部分。

然后我们来编写一个简单的xml:

<?xml version="1.0" encoding="UTF-8"?>

<Window mininfo="200,360" size=" 480,320 ">

<Font name="幼圆" size="16" default="true" /> <VerticalLayout bkcolor="#FFFF00FF"> <Button name="changeskinbtn" height="20" text="测试按钮" maxwidth="120" /> <RichEdit name="testrichedit" bordercolor="#FF0000" bordersize="0"

inset="4,2,4,2" bkcolor="#A0F2F5FA" bkcolor2="#A0FF0000"

font="1" multiline="true" vscrollbar="true" autovscroll="true" borderround="18,18" bkcolor3="#A0F2F5FA"

enabled="true" rich="true" readonly="false" text="测试richedit"> </RichEdit> <Edit name="testedit" text="测试编辑框" /> </VerticalLayout>

</Window>

把以上xml保存为test1.xml,主要保存格式为utf-8(不要使用windows自带的记事本保存,可以使用ultraedit、editplus之类具备xml编辑能力的编辑器保存)。然后运行程序,可以看到如下效果:

好像还不是太难看,不过按钮好像看起来不大像按钮,;然后将button_nor.bmp、button;可以看到按钮的显示已经改变了,我们继续将Rich;继续修改这个xml,我们通过设置xml中控件的属;3响应事件;Duilib中的事件响应有两种方式:;?在事件处理类(一般使用窗口类)中实现INoti;中处理事件,这种方式比较简单常用;classCLoginFr


好像还不是太难看,不过按钮好像看起来不大像按钮,那就给贴个图把,将一下这 行加入到Window标签下: <Default name="Button" value="normalimage="file='button_nor.bmp' corner='4,2,4,2' fade='200' hsl='true'" hotimage="file='button_over.bmp' corner='4,2,4,2' fade='200' hsl='true'" pushedimage="file='button_down.bmp' corner='4,2,4,2' fade='200' hsl='true' " " />

然后将button_nor.bmp、button_over.bmp、button_down.bmp(可在Duilib发行包中找到)放到exe目录下,运行程序,可以看到:

可以看到按钮的显示已经改变了,我们继续将Richedit换个背景,将Richedit的背景色改成bkcolor="#FFF2F5FA" bkcolor2="#FFA0A000" bkcolor3="#FFF2F5FA",我们得到下面的结果:

继续修改这个xml,我们通过设置xml中控件的属性,可以很简单的实现自由调整界面。

3 响应事件

Duilib中的事件响应有两种方式:

? 在事件处理类(一般使用窗口类)中实现INotifyUI接口,然后在Notify函数

中处理事件,这种方式比较简单常用。示例如下:

class CLoginFrameWnd : public CWindowWnd, public INotifyUI

{

public: // ……

void Notify(TNotifyUI& msg)

{

if( msg.sType == _T("click") ) {

if( msg.pSender->GetName() == _T("closebtn") ) { PostQuitMessage(0); return; }

else if( msg.pSender->GetName() == _T("loginBtn") ) { Close(); return; } }

else if( msg.sType == _T("itemselect") ) {

if( msg.pSender->GetName() == _T("accountcombo") ) {

CEditUI* pAccountEdit

static_cast<CEditUI*>(m_pm.FindControl(_T("accountedit"))); =

if( pAccountEdit ) pAccountEdit->SetText(msg.pSender->GetText()); }

}

}

}

? 使用代理机制处理事件

class CLoginFrameWnd : public CWindowWnd, public INotifyUI

{

public:

// …… bool OnAlphaChanged(void* param) {

TNotifyUI* pMsg = (TNotifyUI*)param;

if( pMsg->sType == _T("valuechanged") ) {

m_pm.SetTransparent((static_cast<CSliderUI*>(pMsg->pSender))->GetValue());

} void OnPrepare()

{

CSliderUI* pSilder =

MakeDelegate(this, } return true; static_cast<CSliderUI*>(m_pm.FindControl(_T("alpha_controlor"))); if( pSilder ) pSilder->OnNotify +=

&CFrameWindowWnd::OnAlphaChanged);

}

}

OnPrepare函数需要在控件创建完成之后调用。

4 贴图描述

Duilib的表现力丰富很大程度上得益于贴图描述的简单强大。Duilib的贴图描述分为简单模式和复杂模式两种。

简单模式使用文件名做为贴图描述内容,在这种方式下,此图片将会以拉伸方式铺满控件。

复杂模式使用带属性的字符串表示贴图方式,既支持从文件中加载图片,也可以从资源中加载,具体如下:

? 如果是从文件加载,设置file属性,如file='XXX.png',不要写res和restype属

?

?

?

? 如果从资源加载,设置res和restype属性,不要设置file属性 dest属性的作用是指定图片绘制在控件的一部分上面(绘制目标位置) source属性的作用是指定使用图片的一部分 corner属性是指图片安装scale9方式绘制

?

?

?

? mask属性是给不支持alpha通道的图片格式(如bmp)指定透明色 fade属性是设置图片绘制的透明度 hole属性是指定scale9绘制时要不要绘制中间部分 xtiled属性设置成true就是指定图片在x轴不要拉伸而是平铺,ytiled属性设

置成true就是指定图片在y轴不要拉伸而是平铺:

5 类html文本描述

Duilib使用一种经过简化的类html格式文本来描述复杂的图文格式。使用<>或{}符号来标识语法标签,支持标签嵌套,如<l><b>text</b></l>,但是应该避免交叉嵌套,如<l><b>text</l></b>。

?

?

?

?

?

?

?

?

?

?

?

?

? <b>text</b> 表示text的内容使用粗体 <c #xxxxxx>text</c> 表示text内容使用#xxxxxx颜色,#xxxxxx表示16进制的RGB值 <f x>text</f> 表示text内容使用x序号的字体 <i>text</i> 表示text内容使用斜体 <i x y z> 表示此次插入图片,x表示图片名称,y表示此图片包含几张字图片(可不填,默认值1),z表示当前使用的字图片id(可不填,默认值0) <a x>text</a> 表示text内容有链接功能,x表示链接地址(可不填),用法如app:notepad or http:www.xxx.com,此字符串需要在用户程序中解析处理。 <n> 表示此次换行 <p x>text</p> 表示text内容是一个段落(从这里开始换行),x表示此段落文字水平距离(可不填) <r>text</r> 表示text内容不使用语法标签功能 <s>text</s> 表示text内容被选中(显示选中的背景颜色) <u>text</u> 表示text内容使用下划线 <x i> 表示从此处向后面移动x个像素值 <y i> 表示该行高度固定为y个像素值

使用标签功能需要把控件的showhtml属性设置为true。

6 动态换肤

Duilib是一个以贴图为主要表现手段的界面库,实现换肤非常简单,可以通过给控件设置不同的图片来实现换肤,比如给需要换肤的控件调用CControlUI::SetBkImage。但是针对换肤功能,Duilib提供了更为简单的方法,即使用CPaintManagerUI::ReloadSkin。

假设我们给程序创建了两套皮肤,分别打包成skin1.zip和skin2.zip,在程序运行的时候,执行:

CPaintManagerUI::SetResourceZip(_T("skin2.zip")); // 或者skin1.zip

CPaintManagerUI::ReloadSkin();

这样简单的两行代码,就实现了全部窗口从skin1皮肤到skin2皮肤的切换。你也可以随时再次调用上面两行代码,把皮肤切换回去。

7 Dll插件

? Duilib支持使用外部dll来扩展控件: 在dll中,只需要实现一个接口CreateControl,如

extern "C" __declspec(dllexport) CControlUI* CreateControl(LPCTSTR pstrType)

{

if( _tcscmp(pstrType, _T("ButtonEx")) == 0 ) return new CButtonExUI;

return NULL;

}

? 在使用程序中,需要在WinMain函数把插件dll使用CPaintManagerUI::LoadPlugin

加载进来,然后就可以和内置控件一样使用了。

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPSTR /*lpCmdLine*/, int nCmdShow)

{

CManager::SetInstance(hInstance);

CManager::SetResourcePath(CManager::GetInstancePath());

CManager::LoadPlugin(PLUGINNAME);

// ……

}

8 资源打包

Duilib使用资源打包功能非常简单,在程序开发完成后只需做两个步骤即可:

? 进入资源目录,然后使用具备zip压缩功能的软件(如winrar、winzip、7zip等)把

布局xml和图片等资源压缩到一个zip文件包里。

?在WinMain函数中CPaintManage;CPaintManagerUI::SetReso;intAPIENTRYWinMain(HINST;CManager::SetInstance(hI;CManager::SetResourcePat;CPaintManagerUI::SetReso;//……;这样就完成了资源打包功能,发布程序只需要把exe;


? 在WinMain函数中CPaintManagerUI::SetInstance(hInstance)的后面加入

CPaintManagerUI::SetResourceZip(_T("xxx.zip")),如

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPSTR /*lpCmdLine*/, int nCmdShow)

{

CManager::SetInstance(hInstance);

CManager::SetResourcePath(CManager::GetInstancePath());

CPaintManagerUI::SetResourceZip(_T("xxx.zip"));

// ……

}

这样就完成了资源打包功能,发布程序只需要把exe、dll和zip文件带上就行了。有时为了需要,也可以修改.zip为其他扩展名,比如.dat。

9 Duilib在MFC或WTL程序中的应用

MFC和WTL依托强大的Windows Common Controls实现和十多年的发展,在传统软件界面领域占了统治地位,很多的现有软件代码都是使用MFC(或WTL,下同,不再特别注明)写的。所以对于一个界面库来说,兼容MFC和WTL显的非常重要。

如果一个软件希望从MFC界面转换到Duilib,或者打算使用Duilib做界面,但是又希望使用MFC的其他功能,这时Duilib和MFC混合使用就派上用场了。

在MFC中使用Duilib,可以按以下步骤操作:

? 在BOOL CXXXApp::InitInstance()函数中,添加

杭州月牙儿网络

DuiLib::CManager::SetInstance(CWinAppEx::m_hInstance);

DuiLib::CManager::SetResourcePath(DuiLib::CManager::GetInstancePath()); 注意需要添加在MFC窗口创建之前。

? 创建Duilib窗口类

class CDuiWnd : public DuiLib::CWindowWnd, public DuiLib::INotifyUI {

public:

CDuiWnd();

LPCTSTR GetWindowClassName() const;

UINT GetClassStyle() const;

void Notify(DuiLib::TNotifyUI& msg);

LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam);

public:

DuiLib::CPaintManager m_manager;

};

? 在MFC窗口类中创建Duilib窗口

m_duiWnd.Create(*this, NULL, UI_WNDSTYLE_CHILD, 0, 0, 0, 642, 520); m_duiWnd.Init();

m_duiWnd是MFC窗口类的成员变量。

杭州月牙儿网络

示例:绿色框内是Duilib窗口,外层是MFC对话框。

杭州月牙儿网络

 

12DUILib经典教程(实例)的更多相关文章

  1. 20个网页设计师应该学习的CSS3经典教程实例

    CSS3技术离我们越近,我们也应该学习一些简单的CSS3技术了,而学习最基本的方法就是模仿,以及观看大师作品的案例.收集了20个基础教程,均是涉及到css3应用范围,值得你和我一起共同学习. Smoo ...

  2. WEKA使用教程(经典教程转载)

    http://blog.csdn.net/yangliuy/article/details/7589306 WEKA使用教程(经典教程转载) 标签: lift算法csv数据挖掘class任务 2012 ...

  3. 超多经典 canvas 实例,动态离子背景、移动炫彩小球、贪吃蛇、坦克大战、是男人就下100层、心形文字等等等

    超多经典 canvas 实例 普及:<canvas> 元素用于在网页上绘制图形.这是一个图形容器,您可以控制其每一像素,必须使用脚本来绘制图形. 注意:IE 8 以及更早的版本不支持 &l ...

  4. [转载]HTML5开发入门经典教程和案例合集(含视频教程)

    http://www.iteye.com/topic/1132555 HTML5作为下一代网页语言,对Web开发者而言,是一门必修课.本文档收集了多个HTML5经典技术文档(HTML5入门资料.经典) ...

  5. 《Objective-C开发经典教程》

    <Objective-C开发经典教程> 基本信息 原书名:Beginning Objective-C 原出版社: Apress 作者: (美)James Dovey    Ash Furr ...

  6. 转 Kafka入门经典教程

    Kafka入门经典教程 http://www.aboutyun.com/thread-12882-1-1.html 问题导读 1.Kafka独特设计在什么地方?2.Kafka如何搭建及创建topic. ...

  7. 转载 CSS3 经典教程系列:CSS3 盒阴影(box-shadow)详解

    目标大纲 文章转载 CSS3 经典教程系列:CSS3 盒阴影(box-shadow)详解 IE中CSS-filter滤镜小知识大全 CSS实现跨浏览器兼容性的盒阴影效果

  8. android 教程实例系列

    用户界面部分学起来还真是无处下手哇,总不能一个控件发一篇文吧,略有点费时间啊...这个难道不是边用边学才给力吗..所以我打算从最实用的Button开始下手. 先贴几个链接,好东西: android用户 ...

  9. Android扫盲教程大全经典教程全分享

    Android扫盲教程大全经典教程全分享,相当于android的简单用户手册下载路径 Android扫盲教程大全经典教程全分享.rar

随机推荐

  1. 2019 USP Try-outs 练习赛

    // 好久没更博客了,最近打了很多场练习赛&校内PK赛,大概自闭忙于补题吧 // 9.26 周四练习赛 A. Kolkhozy 题意 有 n 个数 \(f[i]\) ,有 q 次询问(l, r ...

  2. 创建第一个spirngmvc小项目

    题外: 设置目录为源代码目录 1.进入:file->project structure->modules->soures 进入这个里面,选择相应的文件夹.例如src/java里的ja ...

  3. 极限学习机(Extreme Learning Machine)学习笔记

    最近研究上了这个一个东西--极限学习机. 在很多问题中,我大多会碰到两个问题,一个是分类,另一个就是回归.简单来说,分类是给一串数打个标签,回归是把一串数变为一个数. 在这里我们需要处理的数据一般维度 ...

  4. 升级MySQL5.7.22版本_总结记录

    目录 一. mysql5.7安装 0. 背景 1. 准备:下载安装包 2. 安装流程小结 3. 具体步骤 二. mysql5.7的一些变化 一. mysql5.7安装 0. 背景 之前用的5.6版本, ...

  5. 《Java语言程序设计》编程练习6.18(检测密码)

    6.18 (检测密码)一些网站对于密码具有一些规则.编写一个方法,检测字符串是否是一个有效密码.     假定密码规则如下:     • 密码必须至少8位字符.     • 密码仅能包含字母和数字. ...

  6. JDBC_Template(简化代码)

    /** * @Description: TODO(这里用一句话描述这个类的作用) * @Author aikang * @Date 2019/8/27 11:03 */ /* Spring JDBC: ...

  7. Ansible配置及使用

    使用公私钥实现ssh无密码登录 生成公钥和私钥 ssh-keygen -t rsa 公钥相当于锁id_rsa.pub,私钥相当于钥匙id_rsa 借用ssh-copy-id工具实现无密码登录 ssh- ...

  8. shell 通配符,管道符,输入/输出重定向,命令置换

    1. echo 输出   [echo 输出的内容 ]把内容输出到终端上 如果字符串使用双引号,echo命令原样输出   [ echo "hello       world" ]  ...

  9. jvm虚拟内存分布 与 GC算法

    jvm虚拟内存分布 程序计数器(PC寄存器)(线程私有): 每个线程启动的时候,都会创建一个PC(Program Counter,程序计数器)寄存器. PC寄存器的内容总是指向下一条将被执行指令的地址 ...

  10. 多源多汇费用流——poj2516

    网络流的题好难..感觉有点遭不住了 这题用矩阵存图,然后把k个物品,每个物品都求一次费用流 /* 多源多汇的费用流 其实是k个费用流 */ #include<bits/stdc++.h> ...