开发者社区> 老朱教授> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

正在开发一个非.net得数据表格组件,用到.net应用中去……

简介:
+关注继续查看

在家正在开发一个grid COM组件,因为实在用.net Winform提供的DataGrid不爽,功能太少了,搞什么都得自己写大堆代码,连最基本得行选都得自己写datagrid.select(...) ...烦了;还是封装几个COM组件,反正系统也只需要运行到win2k的工控机上;大致确定了一下功能和名称,叫做GridCellPro组件,支持多种功能,比如合计列,合计行,打印,中文金额,过滤,排序,加亮,列排序,固定列,多种CELL;还有一个就是报表,水晶报表很强大,可和我使用习惯不一样,也许是习惯了borland的贴心设计吧,还是做好表格组件再说吧,希望能改善一下开发效率。

目前编写完框架代码,已经把表格框架搞完了,只是属性定制对话框还有点问题,到网上搜来个文章看了看,还不错,关键得地方都讲到了;顺便也收录进来,今后忘了又看。


原始地址:http://hubdog.csdn.net/Hubdog/ActiveX2.htm
转自哈巴狗的小窝

完善ActiveX控件

从VCL出发生成一个基本的ActiveX控件是很容易的,但为了让它更容易使用,比如添加属性页编辑器,还需要做一些额外的工作。接下来我们将继续完善ListX控件,这次不是改进它运行时的功能,而是增强它设计时的编辑功能,为它提供一个属性页和下拉式属性编辑器。

属性列表

如图1.7所示,在Visuabl Basic的窗体上放置一个ListBoxX控件,这时可以注意到在属性察看器中Cursor 和DragCursor 属性,不像通常在Delphi的属性察看器中所看到的ListBox控件的相应属性那样显示crDefault 和其他crDrag 枚举变量,而是显示0和-12的数值。这是因为TCursor 类型在Delphi中是整数类型,所以当ActiveX control wizard转化TListBox 控件时,Cursor属性就按一个整数属性来处理了。这样,在Visual Basic中我们要想设定列表框控件的光标为时钟形状,我们必须设定Cursor 属性为-11, 它等价于定义在Controls单元中的crHourGlass 常数值,但这样显得非常不直观,很容易忘记和混淆。

image001.jpg

图1.7

同时,在Delphi中使用TListBox 控件时,除了是显示一个符号名而不是数值外,属性察看器还会显示一个下拉式列表,里面包含了所有可能的光标值,那么我们能不能使我们的ActiveX控件在Visual Basic中有同样的属性列表呢?Nothing is impossible!我们可以重载TActiveXControl类的GetPropertyString、GetPropertyStrings和GetPropertyValue 方法来做到。下面就是修改后的ListBox控件代码:

unit ListBoxImpl;

interface

uses

Windows, ActiveX, Classes, Controls, Graphics, Menus, Forms, StdCtrls,

ComServ, StdVCL, AXCtrls, DelphiByDesignXLib_TLB;

type

TListBoxX = class( TActiveXControl, IListBoxX )

private

{ Private declarations }

FDelphiControl: TListBox;

//省略…

protected

{ Protected declarations }

procedure InitializeControl; override;

procedure EventSinkChanged(const EventSink: IUnknown); override;

procedure DefinePropertyPages( DefinePropertyPage: TDefinePropertyPage); override;

function GetPropertyString( DispID: Integer; var S: string ): Boolean; override;

function GetPropertyStrings( DispID: Integer; Strings: TStrings ): Boolean; override;

procedure GetPropertyValue( DispID, Cookie: Integer; var Value: OleVariant ); override;

{ Methods that support properties }

. . .

end;

implementation

uses

TabWidthPpg, AboutListBox, SysUtils;

function TListBoxX.GetPropertyString( DispID: Integer;var S: string ): Boolean;

begin

case DispID of

5: // 5 =IlistBoxXDisp接口中DragCursor属性的DispID

begin

S := CursorToString( Get_DragCursor );

Result := True;

end;

26: // 26 = IlistBoxXDisp接口中Cursor属性的DispID

begin

S := CursorToString( Get_Cursor );

Result := True;

end;

else

Result := False;

end;

end;

function TListBoxX.GetPropertyStrings( DispID: Integer; Strings: TStrings ): Boolean;

var

I: Integer;

Cookie: Integer;

TempList: TStringList;

begin

case DispID of

5, // 5 =IlistBoxXDisp接口中DragCursor属性的DispID

26: // 26 = IlistBoxXDisp接口中Cursor属性的DispID

begin

TempList := TStringList.Create;

try

GetCursorValues( TempList.Append );

for I := 0 to TempList.Count - 1 do

begin

Cookie := StringToCursor( TempList[ I ] );

Strings.AddObject( TempList[ I ], TObject( Cookie ) );

end;

finally

TempList.Free;

end;

Result := True;

end;

else

Result := False;

end;

end;

procedure TListBoxX.GetPropertyValue( DispID, Cookie: Integer;

var Value: OleVariant );

begin

case DispID of

5, // 5 =IlistBoxXDisp接口中DragCursor属性的DispID

26: // 26 = IlistBoxXDisp接口中Cursor属性的DispID

begin

{ Cookie 代表被选的项目}

Value := Cookie;

end;

end;

end;

{= 省略…}

initialization

TActiveXControlFactory.Create( ComServer, TListBoxX, TListBox,

Class_ListBoxX, 1, '{B19A64E4-644D-11D1-AE4B-444553540000}', 0);

end.

这里TListBoxX 控件由于是继承于TActiveXControl,所以在TListBoxX类中可以重载上面的方法。

GetPropertyString 函数在属性察看器显示一个属性时会被调用,通过重载这个方法,我们可以为一个属性值提供一个字符串表达。它同Delphi的属性编辑器TPropertyEditor类的GetValue方法比较类似。

GetPropertyString 函数会提供两个参数,第一个是被请求字符串表达的属性的dispid。因为所有控件属性的显示都会调用这个方法,所以必须过滤要处理的属性,这通过用Case语句来检验Dispid就可以了。要想确定每个属性对应的Dispid,可以察看类型库接口单元。类型库单元的代码如下:

unit DelphiByDesignXLib_TLB;

{ This file contains pascal declarations imported from a type library.

This file will be written during each import or refresh of the type

library editor. Changes to this file will be discarded during the

refresh process. }

interface

uses Windows, ActiveX, Classes, Graphics, OleCtrls, StdVCL;

const

LIBID_DelphiByDesignXLib: TGUID =

'{B19A64DB-644D-11D1-AE4B-444553540000}';

const

{ Component class GUIDs }

Class_ListBoxX: TGUID = '{B19A64DE-644D-11D1-AE4B-444553540000}';

type

{ Forward declarations: Interfaces }

IListBoxX = interface;

IListBoxXDisp = dispinterface;

IListBoxXEvents = dispinterface;

{ Forward declarations: CoClasses }

ListBoxX = IListBoxX;

{ Forward declarations: Enums }

TxBorderStyle = TOleEnum;

TxDragMode = TOleEnum;

TxImeMode = TOleEnum;

TxListBoxStyle = TOleEnum;

TxMouseButton = TOleEnum;

{ Dispatch interface for ListBoxX Control }

IListBoxX = interface(IDispatch)

['{B19A64DC-644D-11D1-AE4B-444553540000}']

function Get_DragCursor: Smallint; safecall;

procedure Set_DragCursor(Value: Smallint); safecall;

. . .

function Get_Cursor: Smallint; safecall;

procedure Set_Cursor(Value: Smallint); safecall;

. . .

procedure AboutBox; safecall;

. . .

property DragCursor: Smallint

read Get_DragCursor write Set_DragCursor;

. . .

property Cursor: Smallint read Get_Cursor write Set_Cursor;

end;

{ DispInterface declaration for Dual Interface IListBoxX }

IListBoxXDisp = dispinterface

['{B19A64DC-644D-11D1-AE4B-444553540000}']

property BorderStyle: TxBorderStyle dispid 1;

property Color: TColor dispid 2;

property Columns: Integer dispid 3;

property Ctl3D: WordBool dispid 4;

property DragCursor: Smallint dispid 5;

property DragMode: TxDragMode dispid 6;

property Enabled: WordBool dispid 7;

property ExtendedSelect: WordBool dispid 8;

property Font: Font dispid 9;

property ImeMode: TxImeMode dispid 10;

property ImeName: WideString dispid 11;

property IntegralHeight: WordBool dispid 12;

property ItemHeight: Integer dispid 13;

property Items: IStrings dispid 14;

property MultiSelect: WordBool dispid 15;

property ParentColor: WordBool dispid 16;

property ParentCtl3D: WordBool dispid 17;

property Sorted: WordBool dispid 18;

property Style: TxListBoxStyle dispid 19;

property TabWidth: Integer dispid 20;

property Visible: WordBool dispid 21;

procedure Clear; dispid 22;

property ItemIndex: Integer dispid 23;

property SelCount: Integer readonly dispid 24;

property TopIndex: Integer dispid 25;

property Cursor: Smallint dispid 26;

procedure AboutBox; dispid -552;

end;

{ Events interface for ListBoxX Control }

IListBoxXEvents = dispinterface

['{B19A64DD-644D-11D1-AE4B-444553540000}']

procedure OnClick; dispid 1;

procedure OnDblClick; dispid 2;

procedure OnKeyPress(var Key: Smallint); dispid 3;

procedure OnColorItem(Index: Integer;

var Color: TColor); dispid 4;

end;

implementation

end.

从上面可以看到,与控件属性对应的Dispid可以在IListBoxXDisp 声明部分找到。

GetPropertyString 函数的第二个参数是一个字符串变量。这个字符串变量将会被显示为相应的属性值,但要注意的是函数的返回值一定要设为True,返回False 将使属性察看器忽略返回的字符串参数。

在上面的GetPropertyString 方法中,使用了CursorToString 函数来转化当前属性值为一个字符串,同时Get_Cursor和Get_DragCursor方法被用来获得当前的属性值。

实现GetPropertyString 方法可以确保对于每个Cursor数值都会有一个字符串来说明它,但是如何显示所有光标的下拉列表呢?这时就必须重载GetPropertyStrings 和GetPropertyValue 方法了。GetPropertyStrings 负责生成一个将要显示在下拉列表中的字符串列表。当用户从列表中选择了一个列表项后,GetPropertyValue 方法就会被调用来返回真正的数值。

首先让我们来看GetPropertyStrings 函数,这个方法有两个参数:一个是Dispid,一个是字符串列表,Dispid自然还是对应于操作的属性,而字符串列表则是用来容纳将要显示在下拉列表中的字符串,此外,字符串列表还必须为每个列表项保存一个唯一的“cookie”值,当用户选择了一个列表项后,对应的cookie值会传递给GetPropertyValue 方法。

为了把cookie值同列表中的字符串关联起来,调用Strings.AddObject 方法来添加字符串到列表中:

try

GetCursorValues( TempList.Append );

for I := 0 to TempList.Count - 1 do

begin

Cookie := StringToCursor( TempList[ I ] );

Strings.AddObject( TempList[ I ], TObject( Cookie ) );

end;

finally

image002.jpg

图1.8

把要显示的字符串作为第一个参数,而第二个参数为Cookie值来调用。Cookie值可以取任意值,这里使用的是Cursor本身对应的整数值。注意AddObject方法的第二个参数需要是TObject类型的参数,所以这里进行了一次类型映射。

GetPropertyValue 方法则有三个参数:第一个参数是属性对应的Dispid,第二个参数是对应于被选列表项的Cookie值,而第三个是用来返回属性值的参数。这个方法的处理非常简单,对于我们这个例子来说,只要把Cookie值返回就可以了,因为它就是实际的Cursor对应的值。

实现这些方法后,我们要做的就是重新注册控件了,图1.8显示了新界面更加友好的属性下拉列表。

实现属性页支持

仔细看一下Visual Basic中的ListBoxX控件的属性察看器,就会发现Items 属性没有显示在其中,但ListBoxX控件是有Items 属性的,其实这是因为ActiveX Control Wizard把它由published属性声明为public类型的属性了,这使得我们只能在运行时操作Items 属性,而无法在设计时编辑Items 属性。因为Visual Basic没有为Items类型的属性提供缺省的属性编辑器,所以ActiveX Control Wizard就没有published Items属性。

image003.gif

图1.9

但可以通过自定义的属性页编辑器来提供对Items属性设计时的编辑支持。属性页总能在各种支持ActiveX控件程序中看到,它提供了方便的属性编辑功能。Delphi 提供了一些现成的预定义的属性页支持,可以把它同ActiveX控件相关联。每个属性页有一个对应的Class ID,它声明在AxCtrls单元中,图1.9列出了每个ID和对应的属性页的功能:

这些预定义的属性页被设计成可以用于任意的ActiveX控件。每个属性页都使用运行时类型信息 (RTTI)来确定控件中哪个属性可以使用相应的属性页进行编辑,并会把每个属性名添加到一个下拉编辑框中,这样的话,一个属性页就可以编辑所有相同类型的属性了。图1.10显示了用来编辑ListBoxX控件的Items属性的字符串属性页:

要想把预定义的属性页同ActiveX控件相关联,只要在控件的DefinePropertyPages 方法中添加对DefinePropertyPage方法的调用就可以了,下面代码把字符串和字体属性页同ListBoxX控件进行了关联:

image004.gif

图1.10

{ TListBoxX }

procedure TListBoxX.DefinePropertyPages(

DefinePropertyPage: TDefinePropertyPage );

begin

{ 把预定义的属性页同控件关联 }

DefinePropertyPage( Class_DStringPropPage );

DefinePropertyPage( Class_DFontPropPage );

//省略…

end;

用户定制的属性页

除了使用预定义的属性页外,还可以创建自定义的属性页。选菜单File|New,切换到ActiveX页,然后选择Property Page项,Delphi将生成一个属性页的窗体文件和单元文件。下面就是生成的TabWidthPpg属性页的代码:

unit TabWidthPpg;

interface

uses

SysUtils, Windows, Messages, Classes, Graphics, Controls, StdCtrls, ExtCtrls, Forms,

ComServ, ComObj, StdVcl, AxCtrls,

ComCtrls;

type

TPpgTabWidth = class(TPropertyPage)

GrpPreview: TGroupBox;

GrpTabWidth: TGroupBox;

LstPreview: TListBox;

ChkUseTabs: TCheckBox;

TrkTabWidth: TTrackBar;

procedure ChkUseTabsClick(Sender: TObject);

procedure TrkTabWidthChange(Sender: TObject);

private

{ Private declarations }

protected

procedure UpdatePropertyPage; override;

procedure UpdateObject; override;

public

{ Public declarations }

end;

const

Class_PpgTabWidth: TGUID =

'{8BE91420-9070-11D1-AE4B-44455354616F}';

implementation

{$R *.DFM}

procedure TPpgTabWidth.UpdatePropertyPage;

var

I: Integer;

begin

{ 使用OleObject更新对象}

{ 从对象中复制字符串到预览列表框 }

for I := 0 to OleObject.Items.Count - 1 do

LstPreview.Items.Add(OleObject.Items[I]);

ChkUseTabs.Checked := OleObject.TabWidth > 0;

TrkTabWidth.Position := OleObject.TabWidth div 4;

LstPreview.TabWidth := OleObject.TabWidth;

end;

procedure TPpgTabWidth.UpdateObject;

begin

{ 使用OleObject更新控件TabWidth属性}

OleObject.TabWidth := LstPreview.TabWidth;

end;

procedure TPpgTabWidth.ChkUseTabsClick(Sender: TObject);

begin

TrkTabWidth.Enabled := ChkUseTabs.Checked;

if ChkUseTabs.Checked then

LstPreview.TabWidth := TrkTabWidth.Position * 4

else

LstPreview.TabWidth := 0;

end;

procedure TPpgTabWidth.TrkTabWidthChange(Sender: TObject);

begin

Modified;

LstPreview.TabWidth := TrkTabWidth.Position * 4;

end;

initialization

TActiveXPropertyPageFactory.Create(

ComServer, TPpgTabWidth, Class_PpgTabWidth);

end.

image005.gif

图1.11

注意上面TPpgTabWidth 类是从TPropertyPage继承来的,它的上层父类是TCustomForm。因此我们可以像普通的Delphi窗体一样定制属性页。这个属性页允许用户可视化的调整ActiveX控件的TabWidth 属性。图1.11就是属性页示意图。

属性页显示时应该能够根据存储在ActiveX控件中的数据更新相应属性页页面的显示,这一切都可以在UpdatePropertyPage方法中来加以实现。上面代码中的TPpgTabWidth. UpdatePropertyPage 方法就是先从ActiveX控件中获得字符串,接着设定ListPreview 列表框,然后是checkbox状态和TabWidth 属性被初始化。一旦激活属性页后,必须能够根据用户输入更新ActiveX控件的属性,这是通过实现UpdateObject 方法来完成的。上面代码中TPpgTabWidth.UpdateObject 方法就是简单的根据预览列表的TabWidth设定更新ActiveX控件的。要注意的是方法中用到的OleObject 就代表ActiveX控件,因此可以使用它来设定数据。

同预定义的属性页一样,我们也要把自定义的属性页同ActiveX控件相关联。同前面一样,通过在DefinePropertyPages 方法中添加对DefinePropertyPage的调用就可以了,这回参数就是新建立的属性页的GUID ,下面就是实现代码:

{ 把用户定制的属性页同控件相关联 }

DefinePropertyPage( Class_PpgTabWidth );

分发ActiveX控件

显然分发包中必须包括编译生成的*.ocx 文件,如果ActiveX项目中用到了其他运行时包,我们还需要分发相应的文件。

如果创建ActiveX控件时选择生成设计时的许可文件,还需要分发相应的*.lic文件。

另外,如果创建的ActiveX项目用到了IStrings 接口或预定义的字体、颜色、字符串或图像属性页的话,我们还必须分发标准的VCL类型库,它包括StdVcl32.dll和StdVcl32.tlb类型库文件。这两个文件是被Delphi安装到了Windows的系统目录下(比如,C:\WinNT\System32)。凡是使用了预定义的属性页的控件必须分发StdVcl32.dll,然而如果只使用了IStrings接口的话,可以只分发StdVcl32.tlb类型库。

注册ActiveX的工具Turbo Register Server

要想使ActiveX控件生效,必须更新注册表信息,这可以使用Turbo Register Server (TRegSvr)程序来完成,这个程序是Borland作为一个演示程序提供的。例子源代码位于Demos\ActiveX\TRegSvr目录下。

TRegSvr 是一个简单的命令行程序,它支持一些开关来控制ActiveX控件的注册,除了要注册的文件名外不加任何参数的话就会注册指定文件。加了-u开关的话就会注销ActiveX控件,-t开关是用来表明ActiveX控件的类型库也应该被注册,如果要注册的文件名为*.tlb,则这个开关可以不设定。-q开关表示TRegSvr不显示任何输出信息,这使得它比较适用于嵌入于安装程序中。

下面是使用TRegSvr程序注册TListBoxX 控件的例子示意:

TRegSvr -q XXX.ocx

TRegSvr -q StdVcl32.dll




本文转自suifei博客园博客,原文链接:http://www.cnblogs.com/Chinasf/archive/2005/04/19/140860.html,如需转载请自行联系原作者

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
OmniTool.NET开发包 - 用C#高效开发Omni/USDT区块链应用
OmniTool.Net开发包适用于为.Net/C#应用快速增加对Omni Layer/USDT数字资产的支持能力,即支持使用自有Omni Layer节点的应用场景,也支持基于第三方API服务和离线裸交易的轻量级部署场景。
1583 0
[译] 依赖注入在多模块工程中的应用
总的来说,这不是一篇关于依赖注入的文章,也不是关于我们为什么选择库 X 而不是库 Y 的文章。 相反的,本文从依赖注入的角度介绍了我们对 Plaid 进行模块化实践的主要成果。
1024 0
ASP.net防止SQL注入方法
1、sql注入比较难防,需要替换select,delete等一打字符  其实对于字符型替换再多都没有替换单引号为两个单引号来的好!对于数字型替换再多都没有用,一定要类型转换。
1140 0
基于Ado.Net的日志组件
软件开发,离不开对日志的操作,它可以帮助我们查找和检测问题。好的日志组件可以对于整个系统来说,至关重要 在NaviSoft产品中,日志组件也占有非常重要的份量。如下图所示,是组件的Db表结构设计 图-1 登录日志:记录系统的登录信息,包括登录用户、登录时间、客户端信息 操作日志:记录每个功能操作信息,包括操作时间、内容和结果 异常错误日志:当整个系统 方法日志:记录每个方法的名称、参数、返回值、所在类和DLL等。
928 0
.net mssql自己封装的 备份/还原 数据库方法
       #region 数据库备份 BackupData         ///         /// 数据库备份         ///         /// MapPath转换过的备份文件目录         /// ...
780 0
WPF的“.NET研究”消息机制(一)- 让应用程序动起来
  前言   谈起“消息机制”这个词,我们都会想到Windows的消息机制,系统将键盘鼠标的行为包装成一个Windows Message,然后系统主动将这些Windows Message派发给特定的窗口,实际上消息是被Post到特定窗口所在线程的消息队列,应用程序的消息循环再不断的从消息队列当中获取消息,然后再派发给特定窗口类的窗口过程来处理,在窗口过程中完成一次用户交互。
892 0
一起谈.NET技术,如何成为人尽皆知的C#开发人员
这篇指导最终会教你如何成为你的同事中最受欢迎的人,您将成为他们在休息和工作时间的聊天的英雄,甚至可以说这篇指导能帮助减少你的工作,你能得到你的同事的慷慨帮助,他们想帮你做你的工作。因为你的名声!   1、变量命名可以显示你创造潜力,不要使用标识法和准则来烦扰你自己,这些全都会限制你的灵感,如果您使用一个未知的命名方案,你会得到赞扬,您的同事会尊敬你。
838 0
一起谈.NET技术,ASP.NET4灵活显标记:数据Web控件改进
  长期以来,使用Web控件的开发者都欣赏这些控件所具有的高生产力,但在控制它们生成的标记方面却付出了一定的代价。例如,许多ASP.NET控件会自动地使用元素来包装它们的内容,原因是为了方便布局或样式控制。
851 0
+关注
3545
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载