正在开发一个非.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,如需转载请自行联系原作者

相关文章
|
1月前
mvc.net分页查询案例——DLL数据访问层(HouseDLL.cs)
mvc.net分页查询案例——DLL数据访问层(HouseDLL.cs)
8 0
|
1月前
|
SQL 开发框架 数据可视化
企业应用开发中.NET EF常用哪种模式?
企业应用开发中.NET EF常用哪种模式?
|
15天前
|
搜索推荐 API C#
.NET开源快速、强大、免费的电子表格组件
.NET开源快速、强大、免费的电子表格组件
|
9天前
|
开发框架 前端开发 JavaScript
采用C#.Net +JavaScript 开发的云LIS系统源码 二级医院应用案例有演示
技术架构:Asp.NET CORE 3.1 MVC + SQLserver + Redis等 开发语言:C# 6.0、JavaScript 前端框架:JQuery、EasyUI、Bootstrap 后端框架:MVC、SQLSugar等 数 据 库:SQLserver 2012
|
1月前
|
SQL 数据库
使用ADO.NET查询和操作数据
使用ADO.NET查询和操作数据
9 0
|
1月前
|
数据安全/隐私保护 Windows
.net三层架构开发步骤
.net三层架构开发步骤
11 0
|
1月前
深入.net平台的分层开发
深入.net平台的分层开发
53 0
|
2月前
|
开发框架 前端开发 .NET
福利来袭,.NET Core开发5大案例,30w字PDF文档大放送!!!
为了便于大家查找,特将之前开发的.Net Core相关的五大案例整理成文,共计440页,32w字,免费提供给大家,文章底部有PDF下载链接。
33 1
福利来袭,.NET Core开发5大案例,30w字PDF文档大放送!!!
|
3月前
|
开发框架 前端开发 .NET
ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程
ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程
41 0
|
1月前
|
开发框架 前端开发 .NET
进入ASP .net mvc的世界
进入ASP .net mvc的世界
29 0