VCL的东西原则上可以由它的Owner来负责他的生命周期,建议不要那样做。
作为和delphi类似的rad(rapid application development)工具,c builder的强大功能不仅体现在数据库开发方面,也凸现于应用程序开发上(令人称绝的是这两方面结合得非常好)。仅就应用程序而言,要真正体现c builder的优势,开发出高质量的软件,则在拖拉拽放之外,尚需用到一些进阶技术。如消息处理、dll、ole、线程、sdk编程。c builder在这些方面都或多或少有独到的优势。此外,可以方便地制作自定义控件,也是c builder的一大特色和高级功能。本文将通过制作一个标题棒在窗口左边的对话框控件,来示范一些c builder中关于控件制作和消息处理的概念,同时涉及到一点sdk编程。我们将要制作的是一个对话框,就如同opendialog等一样,一调用其execute()方法,就d出一个如图一所示的窗口。这个窗口的标题棒位于左方,绿色,文字走向为由下而上的90度字形,其功能和一般的标题棒相同,可以将鼠标移至该处来移动该窗口。
首先来完成这个窗口,然后用它来制作对话框控件。
图一
2 利用wm_nchittest消息制作竖直标题的窗口
.wm_nchittest消息
c builder将某些windows消息封装于事件(event)中,但无法囊括所有消息,如wm_nc**** 系列消息。wm_nchittest消息发生于游标(cursor)移动或鼠标按下、释放时,返回值指示目前游标所在位置,如返回hthscroll表示处于水平滚动条内,返回htcaption表示处于标题棒内(参见win32 sdk help)。其参数xpos、ypos分别表示游标的x、y坐标(相对于屏幕左上角),分别对应于lparam的低字和高字。如果拦截wm_nchittest消息,使得当鼠标在窗口左边按下时,人为地将返回值设为htcaption,则系统将以为是在标题棒内,于是将可以移动窗口,完成了标题棒的功能,至于颜色和文字,则与消息无关,将在下面叙述其原理。
.windows消息
消息就是windows *** 作系统送往程序的事件。但事件数以百计, *** 作系统并没有为各个事件设计不同的消息结构,而是以一个一般性的结构来来描述消息,这个结构在c builder中定义为tmessage。另外c builder对常见消息定义了专用结构,二者对等。可以直接将消息转换为专用结构,也可以自行解释tmessage参数。以wm_nchittest消息为例,它的定义如下:
struct twmnchittest
{
cardinal msg
long unused
union
{
struct
{
windows::tsmallpoint pos
long result
}
struct
{
short xpos
short ypos
}
}
}
对照tmessage定义:
struct tmessage
{
cardinal msg
union
{
struct
{
word wparamlo
word wparamhi
word lparamlo
word lparamhi
word resultlo
word resulthi
}
struct
{
long wparam
long lparam
long result
}
}
}
可以发现,tmessage的lparam成员对应twmnchittest的pos成员,就是说以下两行语句
等价:
tpoint pt=tpoint(msg.lparam)//此时msg类型为tmessage
tpoint pt=tpoint(msg.pos)//此时msg类型为twmnchittest
.c builder处理消息的宏
在c builder中自定义消息处理是较为方便的,结合wm_nchittest举例如下:
在窗口类的protected部分加入如下宏定义:
begin_message_map
message_handler(wm_nchittest,tmessage,onnchittest)
end_message_map(tform)
message_handler包含3个参数:wm_nchittest,消息标识,也可以为自定义消息如wm_mymessage,这时只需加一个宏如#define wm_mymessage wm_app 1等;第二个参数tmessage代表消息类型,也可以为符合要求的自定义消息结构类型如tmymsg等,onnchittest为消息处理函数。这样,一旦有wm_nchittest消息传给tform,对该消息的响应就完全交由onnchittest函数处理。onnchittest函数只有一个参数,类型为message_handler中第2个参数的引用,即tmessage 或tmymsg 。
.完成图一的窗口。
开始一个新应用程序(new application),将form1命名为vcform,对应单元文件为vcap.cpp,头文件为vcap.h。vcform的boarderstyle设置为bsnone,其上放置一个位图按钮bitbtn1,caption为 ok,kind为bkok,onclick事件处理函数中加入一句close()。然后在vcap.h的protected部分加入如前所述消息处理宏和函数onnchittest的声明,以处理标题条的拖动功能。为完成标题的着色和文字输出,双击vcform的onpaint事件以定制formpaint函数,具体代码见下面源码。此外为使窗口有立体感,重载虚函数createparams,以修改窗口的风格。完整的vcap.h和vcap.cpp如下:
//vcap.h
#ifndef vcaph
#define vcaph
#include
#include
#include
#include
#include
class tvcform : public tform
{
__published: // ide-managed components
tbitbtn *bitbtn1
void __fastcall formpaint(tobject *sender)
void __fastcall bitbtn1click(tobject *sender)
private: // user declarations
protected:
void __fastcall onnchittest(tmessage msg)
void __fastcall createparams(tcreateparams params)
begin_message_map
message_handler(wm_nchittest,tmessage,onnchittest)
end_message_map(tform)
public: // user declarations
__fastcall tvcform(tcomponent* owner)
}
extern package tvcform *vcform
#endif
//vcap.cpp
#include
#pragma hdrstop
#include "vcap.h"
#pragma package(smart_init)
#pragma resource "*.dfm"
tvcform *vcform
__fastcall tvcform::tvcform(tcomponent* owner)
: tform(owner)
{
}
void __fastcall tvcform::formpaint(tobject *sender)
{
//绘制宽20的绿色标题条
rect rc
setrect( rc,0,0,clientwidth,clientheight)
canvas->pen->color=clgreen
canvas->brush->color=clgreen
canvas->rectangle(0,0,20,clientheight)
//输出旋转文字
char* msg=caption.c_str()
logfont fontrec
memset( fontrec,0,sizeof(logfont))
fontrec.lfheight = -13
fontrec.lfweight = fw_normal
fontrec.lfescapement = 900//旋转角度900x0.1度=90度
lstrcpy(fontrec.lffacename,"宋体")
hfont hfont=createfontindirect( fontrec)
hfont hold=::selectobject(canvas->handle,hfont)
::setrect( rc,0,0,20,clientheight)
::settextcolor(canvas->handle,rgb(255,255,255))
::textout(canvas->handle,3,clientheight-3,msg,lstrlen(msg))
::selectobject(canvas->handle,hold)
::deleteobject(hfont)
}
void __fastcall tvcform::bitbtn1click(tobject *sender)
{
close()
}
void __fastcall tvcform::onnchittest(tmessage msg)
{
tpoint pt
pt.x=loword(msg.lparam)
pt.y=hiword(msg.lparam)
pt =screentoclient(pt)
rect rc
setrect( rc,0,0,20,clientheight)
if (ptinrect( rc,pt))
msg.result = htcaption
else
defaulthandler( msg)
}
void __fastcall tvcform::createparams(controls::tcreateparams params)
{
tform::createparams(params)
params.style |= ws_popup
params.style ^= ws_dlgframe
}
vcform的消息处理已经介绍过,这里再对标题条的绘制作简要说明。由于c builder的tfont没有定义文字旋转旋转的属性,因此用传统的sdk绘图方法。canvas->handle即是代表gdi绘图的hdc。
3 制作对话框控件在开始制作控件之前,先将vcap.cpp中的#pragma package(smart_init)行注释掉。创建控件的步骤是:创建一个单元文件,在其中完成控件的类定义和注册,然后就可以安装了。控件类一般从某个现有类继承导出。制作控件与一般类定义的主要区别在于属性(property)和事件(event),事件也是属性。由属性就带来了属性的存取方法、缺省值、属性编辑器等问题。为简单起见,本控件只涉及到上述一部分概念,但能涵盖控件制作的一般过程。
.开始一个空控件
由于要制作的对话框控件的最小必要功能是一个execute()方法,因此可以从tcomponent类继承。命名控件名为tvcaptiondlg,定义控件的单元文件命名为vcapdlg.cpp,其头文件为vcapdlg.h。用component wizard或手工方法完成如下文件:
//vcapdlg.h
#ifndef vcapdlgh
#define vcapdlgh
#include
#include
#include
#include
class package tvcaptiondlg: public tcomponent
{
private:
protected:
public:
virtual __fastcall tvcaptiondlg(tcomponent *owner)
__published:
}
#endif
//vcapdlg.cpp
#include
#pragma hdrstop
#include "vcapdlg.h"
#pragma package(smart_init)
static inline tvcaptiondlg * validctrcheck()
{
return new tvcaptiondlg(null)
}
namespace vcapdlg //同控件定义单元文件名,首字母大写,其余小写
{
void __fastcall package register()
{
tcomponentclass classes[1]={__classid(tvcaptiondlg)}
registercomponents("mailuo",classes,0)
}
}
__fastcall tvcaptiondlg::tvcaptiondlg(tcomponent * owner)
:tcomponent(owner)
{
}
registercomponents("mailuo",classes,0)指示在控件面板上的mailuo页(没有则创建该页)上生成classes数组包含的所有控件,这里是一个tvcaptiondlg控件。当然此时的tvcaptiondlg控件不具备tcomponent类之外的任何能力。
.将要用到的form文件包含进来
这只需在vcapdlg.cpp的#include "vcapdlg.h"后加入一行#include "vcap.cpp"(vcapdlg.*与vcap.*在同一目录)即可,重申一句:vcap.cpp中的#pragma package(smart_init)行要去掉。将整个vcap.cpp和vcap.h的内容包括在vcapdlg.cpp中也是可以的,这样就用不着vcap.*文件了.即将类vcform的定义与vcapdlg放在一个文件里,反正vcform只不过是vcapdlg要用到的一个类定义罢了。不过这样一来,在生成vcform的实例对象时,上面所说bitbtn1的caption、kind等与缺省值不等的属性都需要运行时设置,因为非缺省属性是保存在.dfm文件里的。这也是使用了form的类常用单独的单元文件保存的原因。
.添加接口属性
这里只提供一个caption属性供控件使用者用于读取或设置对话框的标题。为此只需在类tvcaptiondlg的声明体的private区加入一个ansistring fcaption变量作内部存储用,并在__published
区加入一行:
__property ansistring caption={read=fcaption, write=fcaption}
因为属性类型是ansistring,所以不需专门的属性编辑器处理设计时属性的编辑。另外在设计时该属性值的改变不需引起什么立即的处理和过程,因此存取方法采用最简单的立即存取(read=fcaption,
write=fcaption)。
.添加执行方法
vcaptiondlg的execute()方法的功能是创建一个类vcform的实例对象并模式显示之。这只需如下代码:
void __fastcall tvcaptiondlg::execute()
{
vcform=new tvcform(application)
vcform->caption=caption
vcform->showmodal()
delete vcform
}
其中vcform为vcap.cpp中已声明的tvcform类类型的一个实例变量。相应地在vcapdlg.h里需加入一个execute方法的声明。
另外可以加入一些无关紧要的代码,如tvcaptiondlg的构造函数中加入成员变量的初始化语句等。至此整个控件的制作完成。完整的控件代码如下:
//vcapdlg.h
#ifndef vcapdlgh
#define vcapdlgh
#include
#include
#include
#include
class package tvcaptiondlg: public tcomponent
{
private:
ansistring fcaption
protected:
public:
virtual __fastcall tvcaptiondlg(tcomponent *owner)
virtual void __fastcall execute()
__published:
__property ansistring caption={read=fcaption, write=fcaption}
}
#endif
//vcapdlg.cpp
#include
#pragma hdrstop
#include "vcapdlg.h"
#include "vcap.cpp"
#pragma package(smart_init)
static inline tvcaptiondlg * validctrcheck()
{
return new tvcaptiondlg(null)
}
namespace vcapdlg
{
void __fastcall package register()
{
tcomponentclass classes[1]={__classid(tvcaptiondlg)}
registercomponents("mailuo",classes,0)
}
}
__fastcall tvcaptiondlg::tvcaptiondlg(tcomponent * owner)
:tcomponent(owner)
{
fcaption="mailuo's sample"
}
void __fastcall tvcaptiondlg::execute()
{
vcform=new tvcform(application)
vcform->caption=caption
vcform->showmodal()
delete vcform
}
控件的安装不再赘述。
4 结语
本文旨在演示c builder的控件制作和消息处理、sdk等高级编程技术。以上代码全部在pwin98/c builder 3.0上通过调试。顺便指出,c builder的帮助文档中的creating custom components讲控件制作讲得非常好,是学习编写控件的不可多得的好教程。但其中making a dialogbox a component一篇中有两处小小瑕疵:一是including the form unit中所讲用pragma link vcap.obj的方法是一个相对麻烦的方法,因为需要先将vcap.cpp放入一个无关项目中编译以生成obj文件(或是用命令行编译但要指定参数),不如本文一条#include"vcap.cpp"简单。二是该文档中没有指出对这种自己生成的form,有一个限制就是一定要注释掉#pragma package(smart_init)行,否则安装时虽可生成包文件但控件不能被装上。它举的about对话框的例子中恰好没有这一句。而用ide产生的form一般都是这一句的。
bcb指的是Borland C++ Builder的简称,Borland是公司名,C++ Builder是一款widows下的集成开发工具。vb是Visual Basic的简称,是由美国微软公司于1991年开发的一种可视化的、面向对象和采用事件驱动方式的结构化高级程序设计语言,可用于开发 Windows 环境下的各类应用程序。它简单易学、效率高,且功能强大可以与 Windows 专业开发工具SDK相媲美。在Visual Basic环境下,利用事件驱动的编程机制、新颖易用的可视化设计工具,使用Windows内部的广泛应用程序接口(API)函数,动态链接库(DLL)、对象的链接与嵌入(OLE)、开放式数据连接(ODBC)等技术,可以高效、快速地开发Windows环境下功能强大、图形界面丰富的应用软件系统。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)