TStringGrid 实现下拉框比较常见的思路是在TSringGrid中嵌入一个TComboBox ,
思路虽然简单,但开发起来却很麻烦,而且有时会出现不愿看到的效果。
还有一种更巧的方法,是Delphi未公开的。
先看两个Delphi提供的VCL:
TInplaceEdit 是Delphi 提供的一个未公开的控件,只能内嵌于TCustomGrid
中使用,TStringGrid 单元格的编辑功能就是由该控件实现的。
TInplaceEditList 是TInplaceEdit的子控件,该控件实现了下拉框和带省略号
的按钮的功能,也是只能在TCustomGrid 中使用。
OnEditButtonClick 事件,按钮按下时触发该事件。
OnGetPickListitems 事件,下拉框弹出前触发该事件,在该事件的处
理中可以改变下拉框的内容。
TEditStyle =(esSimple, esEllipsis, esPickList)。
编辑器类型,esSimple 普通类型,esEllipsis 带省略号按钮类型,
esPickList 下拉框类型。
TCustomGrid ,看一下它的两个protected函数:
FInplaceEdit: TInplaceEdit;
FInplaceEdit 是TCustomGrid的编辑器。
Function GetEditStyle(ACol, ARow: Longint): TEditStyle; dynamic;
获取给定单元格的编辑器属性。
Function CreateEditor: TInplaceEdit; virtual;
TCustomGrid 创建编辑器。
TCustomGrid 只对这两个函数作了最基本的实现,其源代码如下:
function TCustomGrid.CreateEditor: TInplaceEdit;
begin
Result := TInplaceEdit.Create(Self);
end;
function TCustomGrid.GetEditStyle(ACol, ARow: Longint): TEditStyle;
begin
Result := esSimple;
end;
TStringGrid 没有改变这两个函数,因此TStringGrid没法用TInplaceEditList
实现下拉框功能 。
实现原理:
TStringGird 第一次进入编辑状态前由CreateEditor 创建编辑器,在编辑器
显示前会访问TCustomGrid.GetEditStyle 函数,并由该函数的返回值决定编
辑器类型。对于TInplaceEditList而言,如果GetEditStype 返回esEllipsis ,
就是具有按钮功能的编辑器,如果返回值是esPickList ,就是下拉框类型的编
辑器,并在显示之前触发OnGetPickListitems事件。因此在TStringGrid的子控
件中重新实现这两个函数,首先使CreateEditor创建的编辑器为TinplaceEditList
型的,然后在GetEditSyle函数中引出OnGetEditStyle事件,在应用环境中决定编
辑的类型。
unit DropListGrid;
interface
uses
Windows, Messages, SysUtils, Classes, Controls, Grids,Forms;
type
TOnGetEditStyle = procedure(ACol, ARow: Integer;var EditStyle:TEditStyle) of Object;
TOnGridEditButtonClick=procedure(Sender:TObject;ACol,ARow:Integer)of Object;
TDropListGrid = class(TStringGrid)
private
{ Private declarations }
FButtonWidth: Integer;
FDropDownRows: Integer;
FOnEditButtonClick: TOnGridEditButtonClick;
FOnGetPickListitems: TOnGetPickListItems;
FOnGetEditStyle: TOnGetEditStyle;
procedure setFButtonWidth(const Value: Integer);
procedure setFDropDownRows(const Value: Integer);
procedure SetFOnEditButtonClick(const Value: TOnGridEditButtonClick);
procedure SetFOnGetPickListitems(const Value: TOnGetPickListItems);
procedure SetOnGetEditStyle(const Value: TOnGetEditStyle);
procedure ButtonClick(Sender: TObject);
procedure GetPickListitems(ACol, ARow: Integer; Items: TStrings) ;
protected //TInplaceEditList
{ Protected declarations }
function CreateEditor: TInplaceEdit;override;
function GetEditStyle(ACol, ARow: Longint): TEditStyle; override;
public
{ Public declarations }
destructor Destroy; override;
published
{ Published declarations }
property ButtonWidth: Integer read FButtonWidth write setFButtonWidth;
property DropDownRows: Integer read FDropDownRows write setFDropDownRows;
property OnEditButtonClick: TOnGridEditButtonClick read FOnEditButtonClick
write SetFOnEditButtonClick;
property OnGetPickListitems: TOnGetPickListItems read FOnGetPickListitems
write SetFOnGetPickListitems;
property OnGetEditStyle:TOnGetEditStyle read FOnGetEditStyle write SetOnGetEditStyle;
end;
procedure Register;
implementation
procedure Register;
begin
RegisterComponents('SelfVcl', [TDropListGrid]);
end;
{ TDropListGrid }
procedure TDropListGrid.ButtonClick(Sender: TObject);
begin
if Assigned(FOnEditButtonClick) then
FOnEditButtonClick(Self,Col,Row);
end;
function TDropListGrid.CreateEditor: TInplaceEdit;
begin
Result := TInplaceEditList.Create(Self);
result.Parent:=Self;
with TInplaceEditList(result)do
begin
Font:=Self.Font;
if FButtonWidth>0 then
ButtonWidth:= FButtonWidth;
if FDropDownRows>0 then
DropDownRows:=FDropDownRows;
OnEditButtonClick:=ButtonClick;
OnGetPickListitems:=GetPickListitems;
end;
end;
destructor TDropListGrid.Destroy;
begin
if InplaceEditor<>nil then
if TInplaceEditList(InplaceEditor).PickListLoaded then
TInplaceEditList(InplaceEditor).PickList.Items.Clear;
inherited;
end;
function TDropListGrid.GetEditStyle(ACol, ARow: Integer): TEditStyle;
begin
result:=inherited GetEditStyle(ACol, ARow);
if Assigned(FOnGetEditStyle) then
FOnGetEditStyle(ACol,ARow,result);
end;
procedure TDropListGrid.GetPickListitems(ACol, ARow: Integer;
Items: TStrings);
begin
if Assigned(FOnGetPickListitems) then
FOnGetPickListitems(ACol,ARow,Items);
end;
procedure TDropListGrid.setFButtonWidth(const Value: Integer);
begin
FButtonWidth := Value;
if InplaceEditor<>nil then
TInplaceEditList(InplaceEditor).ButtonWidth:=Value;
end;
procedure TDropListGrid.setFDropDownRows(const Value: Integer);
begin
FDropDownRows := Value;
if InplaceEditor<>nil then
TInplaceEditList(InplaceEditor).DropDownRows:=Value;
end;
procedure TDropListGrid.SetFOnEditButtonClick(const Value: TOnGridEditButtonClick);
begin
FOnEditButtonClick := Value;
end;
procedure TDropListGrid.SetFOnGetPickListitems(
const Value: TOnGetPickListItems);
begin
FOnGetPickListitems := Value;
end;
procedure TDropListGrid.SetOnGetEditStyle(const Value: TOnGetEditStyle);
begin
FOnGetEditStyle := Value;
end;
end.
==============下面是我生成SelVcl的过程
1、在delphi中新建一个component
2、将以上代码张贴到新建单元中
3、编译生成一个Pas文件在Lib目录下
4、通过install component加入到SelVcl
5、新建一个工程,在SelVcl页中使用该控件
6、设置属性:goEditing:=true;
7、在OnGetEditStyle事件中写EditStyle:= esPickList;
调试可以看到,在点击网格进入编辑状态后,就会显示下列按钮