通过实例看VCL组件开发全过程(四)

四、组件属性编辑器和组件编辑器:

通过上面的努力我们的组件似乎已经比较完美了,可我们也忽略了一些重要的细节和一些有趣的事情,这一篇我们将研究两个很有用的组件特性:

在之前开发组件核心功能时我们曾设置了两个属性BeginTimeWakeTime, 他们都是字符串型的属性,然而他们所要表示的却是时间类型,这样就很有可能使组件使用者错误的编辑属性并导致转化字符串到时间时出错(当然这里只是为了文 章的讲解,我们故意把它设置为了字符串类型),虽然通过浏览原代码你知道我们也做了一些代码级别的防出错处理,使当输入错误时属性自动变成‘00:00:00’,然而这对组件使用者来讲仍然显的很不友好,所以我们需要为这两个属性定制编辑器,我们的编辑器将弹出一个窗口里面有一个TdateTimePicker用来选择时间。在delphi中有许多这样的例子,例如大家都知道的lines属性,当你单击它右放的省略号时为自动弹出一个文本编辑器来编辑lines,这大大降低了组件使用者范错误的可能性。

在定制完属性编辑器以后,我们将为组件本身加入一写有趣的元素——组件编辑器,这也是在delphi中 经常出现的,例如有些组件当你双击它时,它并不会进入代码编写状态,而是弹出它自己的编辑器。虽然我们的组件似乎并不需要这种特性,但为了演示它,我们也 将它考虑近来,我们给我们的组件编写了一个版权信息和一个关于对话框,当组件使用者双击它时弹出关于信息(当然,这仅仅是种演示)。上面提到的两种特性由 于它们只是会在设计时起作用,所以你完全可以在新的组件包中编写并注册它们,并将这个组件包设置为Designtime Only,为了方便起见我们就直接把它们和组件的单元编写在一起。注意:以下出现的一些类和方法都需要引用单元DesignEditorsdelphi7)或DsgnIntfdelphi5),与前面说的一样,它们都属于delphiopen tools api所以,如果你没有这写单元请按照前文的方法安装它们。

首先来编写属性编辑器,由于BeginTimeWakeTime是字符串类型,所以我们必须从默认的字符串属性编辑器类TstringProperty继承并覆盖它的一写方法(这里只介绍几个重要的方法,事实上所有的属性编辑器都从TpropertyEditor继承而来,然而我们不用直接继承这个基类)。其中一个重要的方法是GetAttributes,他将返回一些代表编辑器功能的值,这些值将会在代码的注释中说明(如果你的属性编辑器还需要一个下拉列表,你还需要另外一个重要的方法GetValues具体请查看delphi帮助)另外为了使属性编辑器为弹出的对话框我们需要覆盖Edit方法。为了可以以可视化的方式设计对话框,我们可以建立一个普通工程,在设计好后将窗体的类声明复制到我们的组件单元,并将窗体的dfm文件拷贝到我们的组件包目录,并在代码中加入编译器开关{$R *.dfm}。以下是窗体的类声明,这个窗体没有任何的代码需要编写:

TTimeEditFrm = class(TForm)

DateTimePicker1: TDateTimePicker;

Button1: TButton;

Button2: TButton;

private

{ Private declarations }

public

{ Public declarations }

end;

以下是属性编辑器的代码:

TClockProperty=class(TStringProperty)

public

function GetAttributes:TPropertyAttributes;override;

procedure Edit;override;

end;

实现部分:

procedure TClockProperty.Edit;

var

TimeEditFrm:TTimeEditFrm;

begin

TimeEditFrm:=TTimeEditFrm.Create(Application);

try

TimeEditFrm.DateTimePicker1.Time:=StrToTime(GetValue);

if TimeEditFrm.ShowModal=mrOK then

SetValue(TimeToStr(TimeEditFrm.DateTimePicker1.Time));

//GetValueSetValueTStringProperty的基类方法,他直接读取和设置字符串的值

finally

TimeEditFrm.Free;

end;

end;

function TClockProperty.GetAttributes: TPropertyAttributes;

begin

result:=[paDialog,paMultiselect];

//paDialog表示属性编辑器将显示一个对话框,paMulitiselect允许多个组件选择属性

//除此之外如果你想让属性编辑器显示下拉列表,你还需要paValueList具体请查看帮助

end;

最后我们用RegisterPropertyEditor方法注册属性编辑器:

procedure Register;

begin

……

RegisterPropertyEditor(TypeInfo(string),TClock,'BeginTime',TClockProperty);

RegisterPropertyEditor(TypeInfo(string),TClock,'WakeTime',TClockProperty);

end;

重新编译更新组件后我们就可以测试了,可以看到组件编辑器工作的很好:

<shapetype coordsize="21600,21600" o:spt="75" o:preferrelative="t" path="m@4@5l@4@11@9@11@9@5xe" filled="f" stroked="f" id="_x0000_t75"><stroke joinstyle="miter"></stroke><formulas><f eqn="if lineDrawn pixelLineWidth 0"></f><f eqn="sum @0 1 0"></f><f eqn="sum 0 0 @1"></f><f eqn="prod @2 1 2"></f><f eqn="prod @3 21600 pixelWidth"></f><f eqn="prod @3 21600 pixelHeight"></f><f eqn="sum @0 0 1"></f><f eqn="prod @6 1 2"></f><f eqn="prod @7 21600 pixelWidth"></f><f eqn="sum @8 21600 0"></f><f eqn="prod @7 21600 pixelHeight"></f><f eqn="sum @10 21600 0"></f></formulas><path o:extrusionok="f" gradientshapeok="t" o:connecttype="rect"></path><lock v:ext="edit" aspectratio="t"></lock></shapetype><shape type="#_x0000_t75" style="width: 324pt; height: 96pt;" id="_x0000_i1025"><img o:title="PE" src="http://www.evget.com/zh-CN/article/articlePic/Dev_Image_2003-8-51748050%5B1%5D.jpg" alt=""></shape>

接下来我们来实现组件编辑器:

组件编辑器需要继承TcomponentEditor并覆盖一些重要的方法,GetVerbCount返回设计时组件右键自定义菜单的数目,GetVerb为每一个自定义菜单添加文字,ExecuteVerb为每一个菜单项添加事件,Edit为组件的缺省操作指定事件(即在设计时双击组件),以下是代码:

TClockEditor=class(TComponentEditor)

public

function GetVerbCount:integer;override;

function GetVerb(index:integer):string;override;

procedure ExecuteVerb(index:integer);override;

procedure Edit;override;

end;

实现部分:

procedure TClockEditor.Edit;

begin

ExecuteVerb(1); //默认显示关于

end;

procedure TClockEditor.ExecuteVerb(index: integer);

begin

case index of

//第一个显示名字的菜单什么都不做显示

1:showmessage('hk.barton@2003');

end;

end;

function TClockEditor.GetVerb(index: integer): string;

begin

case index of

0:result:='hk.barton';

1:result:='About Clock';

end;

end;

function TClockEditor.GetVerbCount: integer;

begin

result:=2;//我们显示两条菜单,一个我的名字,一个关于

end;

同样最后我们注册组件编辑器:

procedure Register;

begin

……

RegisterComponentEditor(TClock,TClockEditor);

end;

同样可以看到组件编辑器的测试情况:

<shape type="#_x0000_t75" style="width: 146.25pt; height: 223.5pt;" id="_x0000_i1026"><img o:title="CE1" src="http://www.evget.com/zh-CN/article/articlePic/Dev_Image_2003-8-51748052%5B1%5D.jpg" alt=""></shape> <shape type="#_x0000_t75" style="width: 101.25pt; height: 99.75pt;" id="_x0000_i1027"><img o:title="CE2" src="http://www.evget.com/zh-CN/article/articlePic/Dev_Image_2003-8-51748054%5B1%5D.jpg" alt=""></shape>

文章写到这里也该结束了,虽然写了那么多,然而在组件开发中这仍是一小部分内容,本文只是抛砖引玉的作用,希望对正要进入组件开发的朋友一些启示。为了方便你阅读本文,如果你想要本文所开发的这个组件的全部原文件,请和我联系:

E-mail:hk.barton@sohu.com hekphi@hotmail.com QQ6813489

(全文完)

参考文献:

Marco Cantu 《Mastering Delphi》

ListView 排序 Stringgrid内使用回车键代替Tab键 TListBox内显示分栏 TListBox每一行显示交互的颜色 TMemo内光标位置根据鼠标移动 TMemo自动卷动 TRichEdit卷到特定位置 TRxRichEdit内插入图像 TStringGrid中插入、删除一行 TStringGrid保存和装载 TTreeview控件显示粗体节点 TWebBrowser中调用“查找”对话框 为Listview栏添加双击事件 为工具栏的TToolButton设置新的索引 仅通过Classname创建和管理任意窗体 从RichEdit取Rtf格式 从TListBox中拖放项目到TRichEdit 从一个TRichedit复制格式Rtf文本到其它 使用CustomSort方法排序TListView 使用DBGrid字段队列同步列标题队列 使用Interfaces和TInterfaceList 使用TRichEdit存储大于64K数据 使用代码移动StringGrid的行和列 使用圆形角显示控件 使用类名显示窗体 保存 装载TCheckListbox值 保存和装载TListView 保存和装载TTreeView 允许TDBGrid栏调整大小但防止移动 克隆控件 克隆窗体 列出控件的所有属性和事件 创建可编辑ListBox 删除Listbox的所有选定项目 删除stringlist中重复项目 删除TStringGrid中的列 动画窗体 取ComboBox List句柄 取TListview内所有选定项目 取TShellListView中选定文件的路径 取列举值的名称 在combobox中实现autocomplete 在DBGrid中使用回车键改变到下一个段 在listbox内列出所有目录、文件和驱动器 在Listbox内显示水平滚动条 在memo中实现UNDO 在RichEdit内搜索文本并选择它 在StringGrid内使用Combobox作为编辑器 在StringGrid单元中画不同颜色 在TComboBox中显示栏 在TComboBox内绘制位图 在TDateTimePicker内显示星期几 在TFileListBox内显示多列 在TListBox中创建彩色项目 在TListBox内拖拽 在TListbox内绘制位图 在TListview内执行二进制搜索 在TListView内拖拽多个项目 在TListView列中创建进度条 在TMemo内查看和编辑MS-DOS文本 在TPageControl拖拽Tabs 在TPrintDialog上放置定制控件 在TRichEdit内使用上标和下标 在TRichEdit内使用超链接 在TRichEdit内文本使用不同的下划线样式 在TRichedit内设置段落行距 在TStringGrid内删除一行 在TStringGrid实现OnColumnClick事件 在自己的控件显示信息 复制Listbox项目到剪贴板 失效TTreeview的tooltips 定制TDateTimePicker的格式 居中控件 屏蔽在EditBox内按回车键的都都声 强制在TEdit内输入 排序StringGrid 搜索和替换RichEdit中文本 搜索和选定TListBox的项目 改变TDBGrid的DefaultRowHeight 改变TRichEdit内选定文本的背景颜色 改变TStringGrid内选定单元的颜色 改变标准对话框 改变状态栏字体样式 改变窗体上所有控件的属性 改变进度条颜色 根据StringGrid列内容自动调整大小 根据列排序TStringGrid 检查Stringgrid中指定单元示范选定 检查TMemo能否取消操作 检查TreeView是否完全展开或折叠 检查TStringGrid是否有滚动条 添加接口对象到list 清空StringGrid的所有单元 移动listbox项目 移动TListView项目 移动TRichEdit内光标到指定位置 聚焦TDBGrid某些单元 自动打开TDateTimePicker 自定义Memo边界 获取TRichEdit中鼠标指针下面的字 访问TRadioGroup的控件 调整TComboBox下拉列表的宽度 转换Editbox的首字符为大写 转换TEdit中每个词的首字母为大写 输出TStringGrid到TListView 运行时创建TButtons队列 运行时创建控件 运行时创建菜单项 运行时替换控件 返回TTreeView内字符串路径 防止在TEdit内剪贴、复制、粘贴 防止用户调整TListView栏大小 限制TEdit的输入 隐藏TListView滚动条 隐藏最小化MDI子窗口 颜色Combo Box 验证TEdit中输入的是数字
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值