DevExpress部分使用技巧

KeyLife富翁笔记
作者 : 轻舞肥羊
标题 : DevExpress部分使用技巧(原创)
关键字: DevExpress,cxGrid
分类 : 开发技巧
密级 : 公开
(评分:★★★★ , 回复: 6, 阅读: 5204) »»
本文由 轻舞肥羊 发表于 大富翁论坛
转载请标明出处,谢谢

声明:以下代码请看懂再使用,本人水平有限,发现bug希望提出,让我们共同改进

 

2004-1-10 17:06:00   
发表评语»»»    

2004-1-10 17:08:06    1.扩展cxLookupComboBox,使其支持多列查询的cxLookupComboBoxEx//====================== ========================================================
// Unit Name: cxLookupComboBoxEx
// Author   : ysai
// Date     : 2003
// Purpose : 扩展cxLookupComboBox,cxDBLookupComboBox,使其支持多列过滤
// History :
//    2003-05-28大数据量改进
//    2003-07-07可操作性改进
//    2003-08-20效率改进
//    2003-08-29加入过滤延时
// 注意:
//     限制1,不能再使用Properties.OnChange事件
//     限制2,不能再使用Properties.ListSource.DataSet.OnFilterRecord事件
//     限制3,不能再使用Properties.ListSource.DataSet.Filtered属性
//     其它,最好在设计期设好一切属性,运行期再设置属性可能引发求知错误
//==============================================================================

unit cxLookupComboBoxEx;

interface

uses
SysUtils, Classes, Controls, Windows, Messages,DB,StrUtils,
cxControls, cxContainer, cxEdit, cxTextEdit,
cxMaskEdit, cxDropDownEdit, cxLookupEdit, cxDBLookupEdit,
cxDBLookupComboBox;

type
TcxLookupComboBoxEx = class(TcxLookupComboBox)
private
    //保存要过滤的字段列表
    FFieldList : TList;
    FFindText   : String;
   
    //过滤事件
    procedure _OnFilterRecord(DataSet: TDataSet; var Accept: Boolean);
    //编辑框文字改变事件
    procedure _OnChange(Sender : TObject);
    procedure GetFieldList;
    //延时过滤消息
    procedure WMTimer (var Message: TMessage); message WM_TIMER;
protected
    //下拉表格收回时
    procedure CloseUp(AAccept: Boolean); override;
    //过滤过程
    procedure _FilterListSource;
    //初始化下拉表格事件
    procedure DoInitPopup; override;
public
    constructor Create(AOwner : TComponent); override;
    destructor Destroy; override;
    //更新要过滤的字段列表
    procedure UpdateFilterFields;
published
end;

TcxDBLookupComboBoxEx = class(TcxDBLookupComboBox)
private
    //保存要过滤的字段列表
    FFieldList : TList;
    FFindText   : String;

    //过滤事件
    procedure _OnFilterRecord(DataSet: TDataSet; var Accept: Boolean);
    //编辑框文字改变事件
    procedure _OnChange(Sender : TObject);
    //取得要过滤的字段列表
    procedure GetFieldList;
    //延时过滤消息
    procedure WMTimer (var Message: TMessage); message WM_TIMER;
protected
    //下拉表格收回时
    procedure CloseUp(AAccept: Boolean); override;
    //过滤过程
    procedure _FilterListSource;
    //初始化下拉表格事件
    procedure DoInitPopup; override;
public
    constructor Create(AOwner : TComponent); override;
    destructor Destroy; override;
    //更新要过滤的字段列表
    procedure UpdateFilterFields;
published
end;

procedure Register;

implementation

const
UM_TIMER_FILTER = WM_USER + $101;             //自定义延时消息ID
FILTERTIMER     = 500;                        //延时时间
DROPDOWN_ROWS   = 12;

procedure Register;
begin
RegisterComponents('Dev Express', [TcxLookupComboBoxEx,TcxDBLookupComboBoxEx]);
end;

{ TcxLookupComboBoxEx }

procedure TcxLookupComboBoxEx.CloseUp(AAccept: Boolean);
begin
inherited;
//收起下拉后取消过滤
if Assigned(Properties.ListSource) then
    if Assigned(Properties.ListSource.DataSet) then
      Properties.ListSource.DataSet.Filtered := False;
end;

constructor TcxLookupComboBoxEx.Create(AOwner: TComponent);
begin
inherited;
//默认值
Properties.AutoSelect                 := False;
Properties.DropDownAutoSize           := True;
Properties.DropDownListStyle          := lsEditList;
Properties.DropDownRows               := DROPDOWN_ROWS;
Properties.DropDownSizeable           := True;
Properties.IncrementalFiltering       := False;
Properties.Revertable                 := True;
Properties.OnChange                   := _OnChange;
Properties.UseLeftAlignmentOnEditing := False;
end;

destructor TcxLookupComboBoxEx.Destroy;
begin
//释放过滤字段列表
if Assigned(FFieldList) then FFieldList.Free;
inherited;
end;

procedure TcxLookupComboBoxEx.DoInitPopup;
begin
//取得过滤字段
if Assigned(Properties.ListSource) then
    if Assigned(Properties.ListSource.DataSet) then
    begin
      GetFieldList;
      Properties.ListSource.DataSet.Filtered := False;
    end;
inherited DoInitPopup;
end;

procedure TcxLookupComboBoxEx._FilterListSource;
//过滤字段
begin
if Assigned(Properties.ListSource)
      and Assigned(Properties.ListSource.DataSet) then
try
    Properties.ListSource.DataSet.DisableControls;
    Properties.ListSource.DataSet.Filtered        := False;
    Properties.ListSource.DataSet.OnFilterRecord := _OnFilterRecord;
    if Text <> '' then
    begin
      FFindText := Text;
      if SelLength > 0 then
        FFindText := LeftStr(Text,SelStart);
      Properties.ListSource.DataSet.Filtered := FFindText <> '';
    end;
    Changed;
finally
    Properties.ListSource.DataSet.EnableControls;
end;
end;

procedure TcxLookupComboBoxEx.GetFieldList;
//取得过滤字段列表
var
i           : Integer;
sFieldName : String;
fdTemp      : TField;
begin
if not Assigned(FFieldList) then
begin
    FFieldList := TList.Create;
    for i:=0 to Properties.ListColumns.Count -1 do
    begin
      sFieldName := Properties.ListColumns.Items[i].FieldName;
      if sFieldName = '' then Continue;
      fdTemp := Properties.ListSource.DataSet.FindField(sFieldName);
      if Assigned(fdTemp) then
        FFieldList.Add(Pointer(fdTemp));
    end;
end;
end;

procedure TcxLookupComboBoxEx._OnChange(Sender: TObject);
//设置延时
begin
if Focused and DroppedDown then
begin
    KillTimer(Handle,UM_TIMER_FILTER);
    SetTimer(Handle,UM_TIMER_FILTER,FILTERTIMER,nil);
end;
end;

procedure TcxLookupComboBoxEx._OnFilterRecord(DataSet: TDataSet;
var Accept: Boolean);
//过滤事件
var
s : String;
i : Integer;
begin
s := LowerCase(FFindText);
if (s <> '') and (Properties.ListColumns.Count > 0) then
begin
    Accept := False;
    for i := 0 to FFieldList.Count -1 do
    begin
      Accept := Pos(s,LowerCase(TField(FFieldList[i]).AsString))>0;
      if Accept then Exit;
    end;
end
else
    Accept := True;
end;

procedure TcxLookupComboBoxEx.WMTimer(var Message: TMessage);
//延时更新消息
begin
KillTimer(Handle,UM_TIMER_FILTER);
if Focused and DroppedDown then _FilterListSource;
end;

procedure TcxLookupComboBoxEx.UpdateFilterFields;
//更新要过滤的字段列表
begin
if Assigned(FFieldList) then
begin
    FFieldList.Free;
    FFieldList := nil;
end;
GetFieldList;
end;

{ TcxDBLookupComboBoxEx }

procedure TcxDBLookupComboBoxEx.CloseUp(AAccept: Boolean);
begin
inherited;
//收起下拉后取消过滤
if Assigned(Properties.ListSource) then
    if Assigned(Properties.ListSource.DataSet) then
      Properties.ListSource.DataSet.Filtered := False;
end;

constructor TcxDBLookupComboBoxEx.Create(AOwner: TComponent);
begin
inherited;
//默认值
Properties.AutoSelect                 := False;
Properties.DropDownListStyle          := lsEditList;
Properties.DropDownRows               := DROPDOWN_ROWS;
Properties.DropDownSizeable           := True;
Properties.IncrementalFiltering       := False;
Properties.Revertable                 := True;
Properties.OnChange                   := _OnChange;
Properties.UseLeftAlignmentOnEditing := False;
end;

destructor TcxDBLookupComboBoxEx.Destroy;
begin
//释放过滤字段列表
if Assigned(FFieldList) then FFieldList.Free;
inherited;
end;

procedure TcxDBLookupComboBoxEx.DoInitPopup;
begin
//取得过滤字段
if Assigned(Properties.ListSource) then
    if Assigned(Properties.ListSource.DataSet) then
    begin
      GetFieldList;
      Properties.ListSource.DataSet.Filtered := False;
    end;
inherited DoInitPopup;
end;

procedure TcxDBLookupComboBoxEx._FilterListSource;
//过滤字段
begin
if Assigned(Properties.ListSource)
      and Assigned(Properties.ListSource.DataSet) then
try
    Properties.ListSource.DataSet.DisableControls;
    Properties.ListSource.DataSet.Filtered        := False;
    Properties.ListSource.DataSet.OnFilterRecord := _OnFilterRecord;
    if Text <> '' then
    begin
      FFindText := Text;
      if SelLength > 0 then
        FFindText := LeftStr(Text,SelStart);
      Properties.ListSource.DataSet.Filtered := FFindText <> '';
    end;
    Changed;
finally
    Properties.ListSource.DataSet.EnableControls;
end;
end;

procedure TcxDBLookupComboBoxEx.GetFieldList;
//取得过滤字段列表
var
i           : Integer;
sFieldName : String;
fdTemp      : TField;
begin
if not Assigned(FFieldList) then
begin
    FFieldList := TList.Create;
    for i:=0 to Properties.ListColumns.Count -1 do
    begin
      sFieldName := Properties.ListColumns.Items[i].FieldName;
      if sFieldName = '' then Continue;
      fdTemp := Properties.ListSource.DataSet.FindField(sFieldName);
      if Assigned(fdTemp) then
        FFieldList.Add(Pointer(fdTemp));
    end;
end;
end;

procedure TcxDBLookupComboBoxEx._OnChange(Sender: TObject);
//设置延时
begin
if Focused and DroppedDown then
begin
    KillTimer(Handle,UM_TIMER_FILTER);
    SetTimer(Handle,UM_TIMER_FILTER,FILTERTIMER,nil);
end;
end;

procedure TcxDBLookupComboBoxEx._OnFilterRecord(DataSet: TDataSet;
var Accept: Boolean);
//过滤事件
var
s : String;
i : Integer;
begin
s := LowerCase(FFindText);
if (s <> '') and (Properties.ListColumns.Count > 0) then
begin
    Accept := False;
    for i := 0 to FFieldList.Count -1 do
    begin
      Accept := Pos(s,LowerCase(TField(FFieldList[i]).AsString))>0;
      if Accept then Exit;
    end;
end
else
    Accept := True;
end;

procedure TcxDBLookupComboBoxEx.WMTimer(var Message: TMessage);
//延时更新消息
begin
KillTimer(Handle,UM_TIMER_FILTER);
if Focused and DroppedDown then _FilterListSource;
end;

procedure TcxDBLookupComboBoxEx.UpdateFilterFields;
//更新要过滤的字段列表
begin
if Assigned(FFieldList) then
begin
    FFieldList.Free;
    FFieldList := nil;
end;
GetFieldList;
end;

end.


2004-1-10 17:14:06    2.动态生成TcxGridDBTableView的列及页脚的合计栏var
i   : Integer;
cl : TcxGridDBColumn;
begin
Screen.Cursor := crHourGlass;
cxtvMaster.BeginUpdate;
try
    cxtvMaster.ClearItems;
    cxtvMaster.DataController.Summary.FooterSummaryItems.Clear;
    for i := 0 to cxtvMaster.DataController.DataSet.FieldCount - 1 do
    begin
      cl := cxtvMaster.CreateColumn;
      cl.DataBinding.FieldName :=
          cxtvMaster.DataController.DataSet.Fields[i].FieldName;
      if cxtvMaster.DataController.DataSet.Fields[i] is TNumericField then
      begin
        TNumericField(cxtvMaster.DataController.DataSet.Fields[i])
            .DisplayFormat := '#,##0.00';
        cl.Width := 80;
        with TcxGridDBTableSummaryItem(
            cxtvMaster.DataController.Summary.FooterSummaryItems.Add) do
        begin
          Column := cl;
          FieldName := cl.DataBinding.FieldName;
          Format := '#,##0.00';
          Kind := skSum;
        end;
      end
      else if cxtvMaster.DataController.DataSet.Fields[i] is TStringField then
        cl.Width := 100
      else
        cl.Width := 80;
      cl.HeaderAlignmentHorz := taCenter;
    end; //if
finally
    cxtvMaster.EndUpdate;
    Screen.Cursor := crDefault;
end;
end;


2004-1-10 17:24:32    3.动态生成TcxGridDBBandedTableView的列及页脚的合计栏如果存储过程或SQL返回如下结果集

员工     1月$ 1月¥ 2月$ 2月¥ 合计$ 合计¥
-------------------------------------------
测试员A 200   1658 300 2487 500   4145

将生成如下样式的Grid(页脚没有画出,麻烦,另外设置了显示格式,金额将以#,##0.00的方式显示)
_________________________________________________
| 员工 |    1月     |    2月     |    合计    |
|-----------------------------------------------|
| 员工 | $ | ¥ | $ | ¥ | $ | ¥ |
|-----------------------------------------------|
|测试员A | 200 | 1658 | 300 | 2487 | 500 | 4145 |
-------------------------------------------------

var
i     : Integer;
cl    : TcxGridDBBandedColumn;
begin
Screen.Cursor := crHourGlass;
cxbtvMaster.BeginUpdate;
try
    cxbtvMaster.ClearItems;
    cxbtvMaster.Bands.Clear;
    cxbtvMaster.DataController.Summary.FooterSummaryItems.Clear;
    for i := 0 to cxbtvMaster.DataController.DataSet.FieldCount - 1 do
    begin
      if i = 0 then
        with cxbtvMaster.Bands.Add do
        begin
          Options.HoldOwnColumnsOnly := True;
          Caption := cxbtvMaster.DataController.DataSet.Fields[i].FieldName;
        end;
      if RightStr(
          cxbtvMaster.DataController.DataSet.Fields[i].FieldName,
          1) = '$' then
        with cxbtvMaster.Bands.Add do
        begin
          Options.HoldOwnColumnsOnly := True;
          Caption := Copy(
              cxbtvMaster.DataController.DataSet.Fields[i].FieldName, 1,
              Length(cxbtvMaster.DataController.DataSet.Fields[i].FieldName)
              - 1);
          cxbtvMaster.DataController.DataSet.Fields[i].DisplayLabel := '$';
        end;
      if RightStr(
          cxbtvMaster.DataController.DataSet.Fields[i].FieldName,
          1) = '¥' then
        cxbtvMaster.DataController.DataSet.Fields[i].DisplayLabel := '¥';

      cl := cxbtvMaster.CreateColumn;
      cl.HeaderAlignmentHorz := taCenter;
      cl.Position.BandIndex := cxbtvMaster.Bands.Count - 1;
      cl.DataBinding.FieldName :=
          cxbtvMaster.DataController.DataSet.Fields[i].FieldName;
      if cxbtvMaster.DataController.DataSet.Fields[i] is TNumericField then
      begin
        TNumericField(cxbtvMaster.DataController.DataSet.Fields[i])
            .DisplayFormat := '#,##0.00';
        cl.Width := 80;
        with TcxGridDBBandedTableSummaryItem(
            cxbtvMaster.DataController.Summary.FooterSummaryItems.Add) do
        begin
          Column := cl;
          FieldName := cl.DataBinding.FieldName;
          Format := '#,##0.00';
          Kind := skSum;
        end;
      end
      else if cxbtvMaster.DataController.DataSet.Fields[i] is TStringField then
        cl.Width := 100
      else
        cl.Width := 80;
    end; //if
finally
    cxbtvMaster.EndUpdate;
    Screen.Cursor := crDefault;
end;
end;


2004-1-31 22:36:25    4.自动调整列宽的方法要注意的地方可以用ApplyBestFit实现自动列宽;
不能在BeginUpdate和EndUpdate之间调用这个方法,否则会产生下标越界错误;
在BeginUpdate和EndUpdate中清除/建立列不会产生屏幕闪烁,其它需要长时间更新cxGrid数据的操作最好放在BeginUpdate和EndUpdate执行,并用try包起来.


2004-2-8 15:32:30    DevExpress Bar的动态菜单没有什么说明,细心点应该知道表结构及数据的内容形式
对比了一下代码,用dxBar比用ToolBar+PopupMenu生成动态菜单要简单,只用了一个递归过程

type
//菜单项
PMenuItemInfo = ^TMenuItemInfo;
TMenuItemInfo = record
    ID            : string;
    ParentID      : string;
    Caption       : string;
    Hint          : string;
    LibraryName   : string;
    ProcedureName : string;
    wParam        : Integer;
    lParam        : Integer;
end;

procedure TmgMainForm.BuildMenu;
//生成菜单

procedure SetMenuItemInfo(
      const ADataSet : TDataSet;
      const AItem     : PMenuItemInfo
      );
begin
    AItem.ID            := ADataSet.FieldByName('ID').AsString;
    AItem.ParentID      := ADataSet.FieldByName('ParentID').AsString;
    AItem.Caption       := ADataSet.FieldByName('Caption').AsString;
    AItem.Hint          := ADataSet.FieldByName('Hint').AsString;
    AItem.LibraryName   := ADataSet.FieldByName('LibraryName').AsString;
    AItem.ProcedureName := ADataSet.FieldByName('ProcedureName').AsString;
    AItem.wParam        := ADataSet.FieldByName('wParam').AsInteger;
    AItem.lParam        := ADataSet.FieldByName('lParam').AsInteger;
end;

procedure CreateItemList(
      const ADataSet : TDataSet;
      const AList     : TList;
      const AText     : string
      );
//根据父节点建立子项目列表
var
    m : PMenuItemInfo;
    i : Integer;
begin
    ADataSet.First;
    for i := 0 to ADataSet.RecordCount - 1 do
    begin
      if ADataSet.FieldByName('ParentID').AsString = AText then
      begin
        New(m);
        SetMenuItemInfo(ADataSet,m);
        AList.Add(m);
        ADataSet.Delete;
      end else
        ADataSet.Next;
    end; //for
end;

procedure CreateMenuItems(
      const ADataSet    : TDataSet;
      const AKeyValue   : string;
      const AItemLinks : TdxBarItemLinks;
      const AIndex      : Integer
      );
//建立菜单项
var
    db : TdxBarButton;
    dbs : TdxBarSubItem;
    l   : TList;
    i   : Integer;
    j   : Integer;
    bg : Boolean;
begin
    bg := False;
    j := AIndex;
    l := TList.Create;
    try
      CreateItemList(ADataSet, l, AKeyValue);
      for i := 0 to l.Count - 1 do
      begin
        if ADataSet.Locate('ParentID', PMenuItemInfo(l[i]).ID, []) then
        begin
          //有子项
          dbs         := TdxBarSubItem.Create(dxBar);
          dbs.Caption := PMenuItemInfo(l[i]).Caption;
          dbs.Hint    := PMenuItemInfo(l[i]).Hint;
          dbs.Tag     := Integer(l[i]);
          if AIndex > 0 then
            dbs.ImageIndex := 0;
          with AItemLinks.Add do
          begin
            Item        := dbs;
            Index       := j;
            BeginGroup := bg;
          end; //with
          bg          := False;
          CreateMenuItems(ADataSet, PMenuItemInfo(l[i]).ID, dbs.ItemLinks, 0);
          if dbs.ItemLinks.Count = 0 then
            dbs.Free
          else
            Inc(j);
        end
        else begin
          //无子项
          if not (PMenuItemInfo(l[i]).Caption = '-') then
          begin
            db          := TdxBarButton.Create(dxBar);
            db.Caption := PMenuItemInfo(l[i]).Caption;
            db.Hint     := PMenuItemInfo(l[i]).Hint;
            db.Tag      := Integer(l[i]);
            db.OnClick := MenuItemClick;
            if AIndex > 0 then
              db.ImageIndex := 0;
            with AItemLinks.Add do
            begin
              Item        := db;
              Index       := j;
              BeginGroup := bg;
            end;
            bg          := False;
            Inc(j);
          end
          else begin
            bg := True;
          end; //if bg
        end; //if Locate
      end; //for
    finally
      l.Free;
    end;
end;

var
rsMenus : TDataSet;
begin
rsMenus := mgDMMain.GetMenus; //取得数据集
if Assigned(rsMenus) then
try
    CreateMenuItems(rsMenus, '', dxBar.Bars[0].ItemLinks, 1);
    //一定要刷新一下,否则不更改样式会出错
    dxBar.Bars[0].ItemLinks[0].Visible := False;
    dxBar.Bars[0].ItemLinks[0].Visible := True;
finally
    rsMenus.Free;
end;
end;


2005-1-4 16:34:35    取得TcxLookupComboBox下拉列表中各项的内容Properties.DataController.Values包含了所有内容,如果要取得当前选择行的内容,用以下代码
//cmb:TcxLookupComboBox;
with cmb.Properties.DataController do
    ShowMessage(Values[FindRecordIndexByKey(cmb.EditValue),0]);
//0代表显示的第一列,以Properties.ListColumns为准,可以从这个集合中查找对应的字段名

你可能感兴趣的:(DevExpress部分使用技巧)