Delphi组件编辑器
看到Dev中的cxGrid组件的编辑器很强大,于是很想探究一下,跟踪cxGrid的代码比较麻烦,但原理大概知道一二.首先来研究一下设计器双击cxGrid弹出一个编辑窗体,选择窗体中的一个内容后,属性编辑器中的内容也随着变化.有了这个特性,对于控件中的一些复杂成员(继承于TPersistent类),如果需要对其进行设置,可以简化为在属性编辑器中进行.cxGrid的代码过于复杂,看看同样效果的标准控件TDataSet控件的字段编辑器的实现吧.
在Delphi的源码中找到DBReg单元,可以看到其注册的组件编辑器代码:
RegisterComponentEditor(TDataset, TDataSetEditor);//TDataSetEditor注册为TDataSet的组件编辑器,响应鼠标双击或右键事件
procedure TDataSetEditor.ExecuteVerb(Index: Integer);//双击或右键菜单触发
begin
if Index = 0 then
ShowFieldsEditor(Designer, TDataSet(Component), GetDSDesignerClass);
end;
继续跟踪ShowFieldsEditor函数:
procedure ShowFieldsEditor(Designer: IDesigner; ADataset: TDataset;
DesignerClass: TDSDesignerClass);
var
FieldsEditor: TFieldsEditor;
vShared: Boolean;
begin
FieldsEditor := CreateFieldsEditor(Designer, ADataSet, DesignerClass, vShared); //创建字段编辑器窗体
if FieldsEditor <> nil then
FieldsEditor.Show;//显示
end;
这个函数核心代码是对CreateFieldsEditor的调用,创建出字段编辑器窗体.跟踪进去看到:
function CreateFieldsEditor(Designer: IDesigner; ADataset: TDataset;
DesignerClass: TDSDesignerClass; var Shared: Boolean): TFieldsEditor;
begin
Shared := True;
if ADataset.Designer <> nil then
begin
Result := (ADataset.Designer as TDSDesigner).FFieldsEditor;
end
else
begin
Result := TFieldsEditor.Create(Application);
Result.DSDesignerClass := DesignerClass;
Result.Designer := Designer;
{Result.Form := Designer.Form;}
Result.Dataset := ADataset;
Shared := False;
end;
end;
TFieldsEditor类是继承于TDesignWindow的编辑器类,TForm是其基类.F12键弹出设计界面,在FieldListBox控件的OnClick事件中,可见到如下代码:
procedure TFieldsEditor.AListBoxClick(Sender: TObject);
begin
UpdateSelection;
end;
procedure TFieldsEditor.UpdateSelection;
var
I: Integer;
Field: TField;
ComponentList: IDesignerSelections;
begin
if Active then
begin
ComponentList := TDesignerSelections.Create;
try
with GetActiveListBox do
for I := 0 to Items.Count - 1 do
if Selected[I] then
begin
Field := TField(Items.Objects[I]){Dataset.FindField(Items[I])};
if Field <> nil then ComponentList.Add(Field);
end;
if ComponentList.Count = 0 then ComponentList.Add(Dataset);
except
raise;
end;
Designer.SetSelections(ComponentList);
end;
end;
看到目标了,最终实现属性编辑器进行切换效果的代码是Designer.SetSelections(ComponentList);我们只需要根据设计界面选择的内容,准备好需要在属性编辑器中显示的对象,添加到TDesignerSelections对象中,调用Designer.SetSelections即可.
另外跟踪一下属性编辑器的编辑框工作原理。
Delphi的属性编辑器可以自动根据属性来调整编辑框的类别,如需要输入字符串的情况就生成一个Edit编辑框,而需要设置Boolean型的就生成CheckBox编辑框,对于集合类型的可以展开依次进行设置。这种方式比较灵活,而且运行我们定制控件的属性编辑器,指定编辑方式。
Delphi没有公开源码,可以跟踪EControl的TInspectorList控件源码,其最终也是调用Delphi控件注册的PropertyEditor来模拟Delphi中操作方式。使用Spy++查看EControl的TInspectorList和Delphi的组件属性编辑器,其原理都是一样的,绘制出所有组件的属性图,左边是属性名称,右边是属性的值,从上到下依次绘制出所有的组件属性。根据鼠标点击的位置获取到鼠标下方属性的索引,从而得到对应的属性,并将属性编辑框移到对应的位置。根据控件的属性所注册的编辑器类型生成属性编辑框的外观,如文本输入框、下拉选择框、复选框等等。
打开EControl的edcPropEdit单元,查看UpdateEditState函数,可看到如下代码:
atr := GetAttributes;
if F_ReadOnly then EditStyle := ieSimple
else if (paValueList in atr) then EditStyle := iePickList
else if (paDialog in atr) then EditStyle := ieEllipsis
else EditStyle := ieSimple;
看到属性的GetAttributes;方法调用了,这是我们下注册属性编辑器的时候指定属性编辑方式的,可以指定弹出对话框、下拉列表等等方式。这里根据注册的编辑方式修改了属性编辑框的外观,如果是paValueList则EditStyle设置为iePickList,属性编辑框的外观生成一个向下的黑色箭头;EditStyle为ieEllipsis的则在编辑框的后面绘制一个带...的按钮。
请见ecExtEdit的PaintBtnGlyph函数:
procedure TCustomEditEx.PaintBtnGlyph(Canvas: TCanvas; Rect: TRect);
const
LeftOffs = 3;
var
Flags: Integer;
W, G, I: Integer;
DC: HDC;
begin
DC := Canvas.Handle;
Flags := 0;
case FEditStyle of
iePickList:
begin
if FActiveList = nil then Flags := DFCS_INACTIVE
else if FPressed then Flags := DFCS_FLAT or DFCS_PUSHED;
DrawFrameControl(DC, Rect, DFC_SCROLL, Flags or DFCS_SCROLLCOMBOBOX); //绘制DFCS_SCROLLCOMBOBOX风格的向下箭头
end;
ieEllipsis:
begin
W := 2;
G := (FButtonWidth - LeftOffs * 2 - 3 * W) div 2;
if G <= 0 then G := 1;
if G > 3 then G := 3;
Flags := Rect.Left + (FButtonWidth - 3 * W - 2 * G) div 2 + Ord(FPressed);
I := Rect.Top + (ClientHeight - W) div 2 {+ Ord(FPressed)};
PatBlt(DC, Flags, I, W, W, BLACKNESS);
PatBlt(DC, Flags + G + W, I, W, W, BLACKNESS);
PatBlt(DC, Flags + 2 * G + 2 * W, I, W, W, BLACKNESS);
end;
end;
end;
对于下拉框风格的属性编辑器来说,下拉框的值是如何填充的呢?打开edcPropEdit单元,查看DropDown函数:
procedure TCustomPropertyEdit.DropDown;
{$IFDEF EC_VCL9_UP}
var wp: IWideProperty10;
{$ENDIF}
begin
if FPropertyEditor <> nil then
begin
PickList.Items.Clear;
{$IFDEF EC_VCL9_UP}
FWListItems.Clear;
TWideStringList(FWListItems).Sorted := PickList.Sorted;
if Supports(FPropertyEditor, IWideProperty10, wp) then
wp.GetValues(GetWStr)
else
{$ENDIF}
FPropertyEditor.GetValues(GetStr);
end;
inherited;
end;
核心代码是FPropertyEditor.GetValues(GetStr); 其中的GetStr是一个函数名称,定义为:
procedure TCustomPropertyEdit.GetStr(const S: string);
begin
PickList.Items.Add(S);
end;
由属性编辑器在获取到值后自动调用GetStr将获取到的内容填充到PickList列表中.而属性编辑器是在UpdateEditor函数中进行设置的,鼠标在点击一个属性的时候会调用UpdateEditor函数。
---------------------
作者:henreash
来源:CSDN
原文:https://blog.csdn.net/henreash/article/details/7371897
版权声明:本文为博主原创文章,转载请附上博文链接!
Delphi组件编辑器的更多相关文章
- delphi 组件容器TObjectList代替List
delphi 组件容器TObjectList代替List TObjectList objList->delete(0); 这个会释放第0行元素的对象 class TTabFormInfo { p ...
- Delphi组件开发教程指南目录
用Delphi开发的时间也不短了!刚接触Delphi的时候,就觉得组件这个东西非常方便!根本不必知道组件内部的作为,只要使用就好了!然而用的多了,也不免会对各种delphi组件的内部实现方式产生兴趣! ...
- delphi 组件安装工具开发
当一个组件的dpk文件数量较多且安装工具不顺手的时候,写一个属于自己的组件安装工具就很有必要了. 本例以 Dev Express 16.1.2 为例,设计一个组件安装工具,以便更深入理解 delphi ...
- 删除delphi组件TStringlist中的重复项目
https://blog.csdn.net/ozhy111/article/details/87975663 删除delphi组件TStringlist中的重复项目 2019年02月27日 15:41 ...
- delphi 组件安装教程详解
学习安装组件的最好方法,就是自己编写一个组件并安装一遍,然后就真正明白其中的原理了. 本例,编写了两个BPL, dclSimpleEdit.bpl 与 SimpleLabel.bpl ,其中,dc ...
- Delphi 组件渐进开发浅谈(二)——双简合璧
2.双简合璧2.1.带有T[x]Label的T[x]Edit组件 请允许我用[x]的书写方式来表示不同的对象.因为随后将大量提及TLabeledEdit与TTntLabeledEdit.TCustom ...
- Delphi 组件渐进开发浅谈(一)——由简入繁
最近业余时间在写游戏修改器玩,对于Delphi自带的组件总觉得差强人意,需要书写大量冗余代码,如果大量使用第三方组件,在以后的移植和与他人交互时也不是很方便,因此便产生了自己封装组件的想法. 实际上这 ...
- Delphi组件indy 10中IdTCPServer修正及SSL使用心得
indy 10终于随着Delphi2005发布了,不过indy套件在我的印象中总是复杂并且BUG不断,说实话,不是看在他一整套组件的面子上,我还是喜欢VCL原生的Socket组件,简洁,清晰.Indy ...
- Delphi组件开发-在窗体标题栏添加按钮(使用MakeObjectInstance(NewWndProc),并处理好多消息)
这是一个在窗体标题栏添加自定义按钮的组件(TTitleBarButton)开发实例,标题栏按钮组件TTitleBarButton以TComponent为直接继承对象,它是一个可以在窗体标题栏上显示按钮 ...
随机推荐
- kotlin递归&尾递归优化
递归: 对于递归最经典的应用当然就是阶乘的计算啦,所以下面用kotlin来用递归实现阶乘的计算: 编译运行: 那如果想看100的阶乘是多少呢? 应该是结果数超出了Int的表述范围,那改成Long型再试 ...
- 简单理解TCP/IP协议
一.什么是TCP/IP TCP/IP是一个协议族,是因为TCP/IP协议包括TCP.IP.UDP.ICMP.RIP.TELNETFTP.SMTP.ARP.TFTP等许多协议,这些协议一起称为TCP/I ...
- Javac可以编译,Java显示找不到或无法加载主类
运行时候加入完整包名.
- 移动/Web开发必备工具!DevExtreme v19.1.7火热发布
DevExtreme Complete Subscription是性能最优的 HTML5,CSS 和 JavaScript 移动.Web开发框架,可以直接在Visual Studio集成开发环境,构建 ...
- SSM项目无法解析JSP页面
JSP页面显示标头<%@ page language="java" contentType="text/html; charset=UTF-8" page ...
- vue 图片滑动登录
前言 最近在研究图片滑动解锁 登录,说是要用阿里的那个验证,但是还是想自己手写下这个Demo 效果图是这样的: 本来是想用canvas 来实现的,但是类,后来还想用css 和图片来代替canvas 其 ...
- Vue 页面 前进刷新 后退不刷新(keepAlive)
前言 遇到这一个个问题 需要是这样的 Vue里面的不刷新问题 页面分为: A 主页 B列表页 C 详情页 A beforeRouteLeave 时设置 to.meta.keepAlive = ...
- 【原】spring+mybatis下sqlSession.delete和insert返回值-2147482646问题
这是由于spring-beans.xml中的batch批处理配置所导致的,注释掉BATCH配置的代码就可以返回1了: <bean id="sqlSessionFactory" ...
- 『HGOI 20190917』Cruise 题解 (计算几何+DP)
题目概述 在平面直角坐标系的第$1$象限和第$4$象限有$n$个点,其中第$i$个点的坐标为$(x_i,y_i)$,有一个权值$p_i$ 从原点$O(0,0)$出发,不重复的经过一些点,最终走到原点, ...
- jdk中使用的设计模式
在JDK(Java Development Kit)类库中,开发人员使用了大量设计模式,正因为如此,我们可以在不修改JDK源码的前提下开发出自己的应用软件,研究JDK类库中的模式实例也不失为学习如何使 ...