亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Delphi > 正文

通用Delphi數據庫輸入控件DBPanel的實現

2019-11-18 19:00:35
字體:
來源:轉載
供稿:網友
李曉平/河北固安華北石油職工大學

---- 無論是開發什么樣的程序,數據輸入是不可缺少的??焖俚厣梢粋€美觀的輸入界面無疑會大大提高程序開發的效率。系統原有的控件,往往不盡人意。在delphi中,如果針對某字段,可選的控件有DBLabel, DBEdit等;如果針對全表的輸入,有DBGrid。使用Dbedit等控件時,用戶必須全面安排各字段的位置,雖然能夠達到美觀的效果,但是,如果字段數多,無疑是很麻煩的。如果采用DBGrid,無論多少個字段,只用一個控件就夠了,簡單倒是簡單,但是各字段是一字排列的,使用起來有點不方便。對于一般的用戶來說,采用表格形式的錄入,既方便,又美觀。這就是本文所要解決的問題。

---- 技術關鍵

---- 本控件的主要功能是實現對數據庫字段的編輯。按照一般的規律,控件中應包含TdataLink對象,還應該實現與TdataLink相關得一系列方法;但是,那樣會耗費大量的代碼。代碼量越大,系統就越復雜,出錯的可能性就越大。本控件的開發思路是以最少的代碼實現最多的功能。所以,對數據字段的編輯直接使用TDBComboBox控件。

---- 為了實現通用性,在控件內部維護了一個字段編輯控件數組和字段標題數組。如下:

Editors: array of TDBComboBox;    
- >具體進行編輯所用的數據控件數組,動態生成
Labels: array of TLabel;        
- >各字段的標題,動態生成

---- 采用TDBComboBox優點是它不僅能具有一般的編輯功能,還能為各字段添加相應的提示信息。代碼如下:
{ 為第I字段增加提示信息的方法}
PRocedure TDBPanel.AddHits
(ItemIndex: Integer; Hits: array of string);
var
  m,n,i: Integer;
begin
  n := Length(Editors);
  m := Length(Hits);
  if ItemIndex< n then begin
    for i:=0 to m-1 do Editors[ItemIndex].Items.Add(Hits[i]);
  end;
end;

---- 具體的應用是千差萬別的,所以,控件還需要給程序員留有足夠的事件處理接口,以實現具體應用時的特殊功能。這就需要在控件中定義一定的事件處理方法供用戶實現。這里提供的是一個OnOkClick事件,即當所有字段編輯完成后所執行的處理。代碼如下:
   
OkButton: TButton;                
- >最后增加的確定按鈕,用于實現提交動作。
property OnOkClick: TNotifyEvent read FClick write FClick;

---- 通過實現OnOKClick方法,用戶可以完成提交、數據合理性檢驗等各種處理工作。另外一個需要特殊處理的是控制在各個字段間的轉換。缺省的情況是用鼠標點擊。但是,用戶的習慣往往是用鍵盤中的"上、下、左、右"四個箭頭鍵。要實現這一功能需定義以下兩個方法:
   
procedure AKeyPress(Sender: TObject; var Key: Char);
procedure AKeyDown(Sender: TObject;
var Key: Word; Shift: TShiftState);

---- 將以上兩個方法賦值給動態生成的Editors,從而實現對箭頭鍵的響應。
---- 不同的表字段數不同,有可能出現顯示不下的情況,這就需要有滾動的功能。所以,在控件中插入了一個TscrollBox控件。最后一個需要注意的是動態控件的撤消及內存的釋放??丶到M的撤消及內存的釋放是有順序的--與創建完全相反的順序。否則會出錯。

----控件的使用

---- 先將DBPanel控件放在窗體上,然后設置數據源屬性、數據輸入表格的列數等屬性。在程序中,打開數據源后,調用創建數據編輯控件的方法即可。即:

Query1.Open;- >打開數據源
DBPanel1.CreateEditors; - >創建各字段的編輯控件
DBPanel1.AddHits(0,['1111','11222','eeee']);    
- >為某字段設置提示信息
DBPanel1.AddHits(1,['1111','11222','eeee']); 
- >為某字段設置提示信息
該控件及示例程序在Win98+Delphi 5.0環境下調試通過。


---- 附件:TDBPanel的源代碼

unit DBPanel;
interface
uses
Windows, Messages, SysUtils, Classes,
Graphics, Controls, Forms, Dialogs,
  ExtCtrls, dbctrls, stdctrls, db;
type
  TDBPanel = class(TPanel)
  private
    { Private declarations }
    FLeft: Integer;
    FTop: Integer;
    maxTextLen: Integer;
    maxLabelLen: Integer;
    FScrollBox: TScrollBox;        {滾動控件}
    FLineHeight: Integer;
    FClick: TNotifyEvent;
    Editors: array of TDBComboBox;    
- >具體進行編輯所用的數據控件數組,動態生成
    Labels: array of TLabel;        
- >各字段的標題,動態生成
    OkButton: TButton;                
- >最后增加的確定按鈕,用于實現提交動作。
    { 數據源}
    FDataSource: TDataSource;
    FColumns: Integer;                
- >輸入表格的列數
    protected
    { Protected declarations }
    procedure FreeEditors;            
- >釋放數據輸入控件的內存
    public
    procedure CreateEditors;//
    (DS: TDataSource; ColCount: Integer);
- >創建各字段的數據輸入控件
    constructor Create(AOwner:
    TComponent); override;
    destructor Destroy; override;
    procedure AKeyPress(Sender:
    TObject; var Key: Char);
    procedure AKeyDown(Sender:
    TObject; var Key: Word; Shift:
    TShiftState);
    procedure ClearHits(ItemIndex: Integer);
    procedure AddHits(ItemIndex:
    Integer; Hits: array of string);
    function Editor(Index: Integer):
    TDBComboBox;
    { Public declarations }
    published
    property LeftLimit: Integer read
    FLeft write FLeft default 10;
    property TopLimit: Integer read
    FTop write FTop default 10;
    property EditorLen: Integer read
    maxTextLen write maxTextLen;
    property LabelLen: Integer read
    maxLabelLen write maxLabelLen    default 100;
    property LineHeight: Integer read
    FLineHeight write FLineHeight    default 15;
    property OnOkClick: TNotifyEvent
    read FClick write FClick;
    property DataSource: TDataSource
    read FDataSource write    FDataSource;
- >數據源
    property Columns: Integer read
    FColumns write FColumns;- >表列數
    { Published declarations }
  end;

procedure Register;

implementation

procedure Register;
begin
  RegisterComponents('Additional', [TDBPanel]);
end;

{ 為第I字段增加提示信息的方法}
procedure TDBPanel.AddHits(ItemIndex:
Integer; Hits: array of string);
var
  m,n,i: Integer;
begin
  n := Length(Editors);
  m := Length(Hits);
  if ItemIndex< n then begin
    for i:=0 to m-1 do Editors[ItemIndex].Items.Add(Hits[i]);
  end;
end;

procedure TDBPanel.AKeyDown
(Sender: TObject; var Key: Word;
  Shift: TShiftState);
begin
  if (Sender is TDBComboBox) then begin
    case Key of
      VK_Next: (Sender as TDBComboBox)
      .DataSource.DataSet.Next;
      VK_PRIOR: (Sender as TDBComboBox)
      .DataSource.DataSet.Prior;
    end;
  end;
end;

procedure TDBPanel.AKeyPress(Sender: TObject; var Key: Char);
begin
  if (Sender is TDBComboBox) then begin
if Key=#13 then (Owner as TForm).Perform(WM_NEXTDLGCTL, 0, 0);
  end;
end;

procedure TDBPanel.ClearHits(ItemIndex: Integer);
var
  n: Integer;
begin
  n := Length(Editors);
  if ItemIndex< n then Editors[ItemIndex].Items.Clear;
end;

constructor TDBPanel.Create(AOwner: TComponent);
begin
  Inherited Create(AOWner);
  FLeft :=10;
  FTop := 10;
  maxTextLen := 100;
  maxLabelLen := 100;
  FLineHeight := 15;
end;

{ 創建各字段的數據輸入控件的方法}
procedure TDBPanel.CreateEditors;//
(DS: TDataSource; ColCount: Integer);
var
  i, n, RowCount: Integer;
  TextHeight: Integer;
begin
  if DataSource.DataSet.Active then begin
    n := DataSource.DataSet.FieldCount;
    { 計算最大的標題長度及顯示長度}
    DataSource.DataSet.First;
    { 計算高度}
    TextHeight := Canvas.TextHeight(DataSource
    .DataSet.Fields[0].DisplayLabel) + FLineHeight; //10;
    { 計算行列數}
    RowCount := n div Columns;
    if n mod Columns <  > 0 then inc(RowCount);
    { 分配內存}
    FreeEditors;
    SetLength(Editors, n);
    SetLength(Labels, n);
    { 創建滾動盒}
    FScrollBox := TScrollBox.Create(Owner);
    FScrollBox.Parent := Self;
    FScrollBox.Align := alClient;
    { 創建編輯}
    for i:=0 to n-1 do begin
      { 創建標題}
      Labels[i] := TLabel.Create(Owner);
      Labels[i].Parent := FScrollBox; //Self;
Labels[i].Caption := DataSource.DataSet.Fields[i].DisplayLabel;
      Labels[i].Left := FLeft + (maxLabelLen +
      maxTextLen + 10) * (i div RowCount);
      Labels[i].Width := maxLabelLen;
Labels[i].Top := FTop + (i mod RowCount) * TextHeight + 5;
      { 創建編輯對象}
      Editors[i] := TDBComboBox.Create(Owner);
      Editors[i].Parent := FScrollBox; //Self;
      Editors[i].Left := Labels[i].Left + Labels[i].Width;
      Editors[i].Width := maxTextLen;
      Editors[i].Top := FTop + (i mod RowCount) * TextHeight;
      Editors[i].DataSource := DataSource;
Editors[i].DataField := DataSource.DataSet.Fields[i].FieldName;
      Editors[i].OnKeyPress := AKeyPress;
      Editors[i].OnKeyDown := AKeyDown;
    end;
    { 創建Ok按鈕}
    OkButton := TButton.Create(Owner);
    OkButton.Parent := FScrollBox;
    OkButton.Left := Editors[n-1].Left;
    OkButton.Top := Editors[n-1].Top + TextHeight;
    OkButton.Caption := '確定';
    OKButton.OnClick := FClick;
  end;
end;

destructor TDBPanel.Destroy;
begin
  FreeEditors;
  Inherited Destroy;
end;

function TDBPanel.Editor(Index: Integer): TDBComboBox;
begin
  if Index< Length(Editors) then Result := Editors[Index]
  else Result := nil;
end;

procedure TDBPanel.FreeEditors;
var
  i,n: Integer;
begin
  { 內存的釋放是要有順序的!必須以創建的相反的順序進行!
  尤其是當組件之間有父子關系時}
  if OkButton<  >nil then OkButton.Free;
  if Editors<  >nil then begin
    n := Length(Editors);
    for i:=0 to n-1 do Editors[i].free;
    Editors := nil;
    n := Length(Labels);
    for i:=0 to n-1 do Labels[i].Free;
    Labels := nil;
  end;
  if FScrollBox<  >nil then begin
    FScrollBox.Free;
    FScrollBox := nil;
  end;
end;

end.

上一篇:用Delphi開發屏幕保護預覽程序

下一篇:Delphi編程實現Ping操作

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
學習交流
熱門圖片

新聞熱點

疑難解答

圖片精選

網友關注

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日本免费一区二区三区视频观看| 欧美国产欧美亚洲国产日韩mv天天看完整| 亚洲成色777777在线观看影院| 69**夜色精品国产69乱| 成人黄在线观看| 精品美女久久久久久免费| 亚洲aⅴ日韩av电影在线观看| 在线播放亚洲激情| 亚洲人成网站在线播| 成人久久精品视频| 日韩免费看的电影电视剧大全| 亚洲国产第一页| 精品亚洲va在线va天堂资源站| 久久夜精品香蕉| 亚洲字幕一区二区| 最近2019年中文视频免费在线观看| 欧美尺度大的性做爰视频| 国自产精品手机在线观看视频| 亚洲美女av在线播放| 欧美日韩电影在线观看| 亚洲欧美激情另类校园| 久久综合网hezyo| 日本亚洲精品在线观看| 欧美美最猛性xxxxxx| 九九热视频这里只有精品| 亚洲欧美中文日韩v在线观看| 国内成人精品一区| 国产视频久久久久久久| 91成人福利在线| 欧美日本高清视频| 国产亚洲激情视频在线| 日本一区二区在线免费播放| 久久天堂av综合合色| 清纯唯美日韩制服另类| 九九九热精品免费视频观看网站| 国产精品久久久一区| 日韩中文字幕在线| 久久人人爽人人爽爽久久| 午夜美女久久久久爽久久| 亚洲精品福利视频| 欧美巨猛xxxx猛交黑人97人| 国产欧美欧洲在线观看| 国产精品第二页| 久久99久久99精品免观看粉嫩| 成人精品在线观看| 欧美成人四级hd版| 91最新国产视频| 日韩在线视频中文字幕| 国产一区二区三区三区在线观看| 日韩av第一页| 26uuu另类亚洲欧美日本老年| 国产91露脸中文字幕在线| 欧美日韩免费看| 成人精品在线视频| 久久久久久综合网天天| 久久亚洲国产成人| 日韩在线国产精品| 欧美日韩一区二区免费在线观看| 国产精品偷伦视频免费观看国产| 成人免费福利视频| 亚洲欧美国产一本综合首页| 国产精品福利在线观看网址| 色婷婷综合成人| 91综合免费在线| 欧美精品一二区| 日韩在线观看免费av| 久久久久久国产| 精品久久久精品| 亚洲国产成人av在线| 久久免费观看视频| 欧美激情乱人伦| 亚洲精品国产精品国自产在线| 亚洲国产精品大全| 亚洲电影成人av99爱色| 亚洲在线www| 日韩av电影国产| 欧美肥臀大乳一区二区免费视频| 色婷婷**av毛片一区| 2018日韩中文字幕| 久久久久久国产| 欧美亚洲另类在线| 久久免费在线观看| 久久综合网hezyo| 狠狠躁夜夜躁久久躁别揉| 精品国产老师黑色丝袜高跟鞋| 91久久精品美女| 精品成人av一区| 亚洲天堂男人的天堂| 欧美成人午夜激情在线| 欧美肥臀大乳一区二区免费视频| 国产女人精品视频| 国产精品直播网红| 在线播放日韩av| 欧美性xxxx极品高清hd直播| 黄色一区二区在线观看| 国产精品久久久久久久天堂| 色妞在线综合亚洲欧美| 日韩有码在线电影| 中文在线不卡视频| 欧美激情视频播放| 色与欲影视天天看综合网| 欧美性极品xxxx做受| 国产成人精品日本亚洲| 91免费人成网站在线观看18| 97精品国产97久久久久久| 日韩国产精品一区| 亚洲精品v天堂中文字幕| 亚洲色图综合久久| 久久久久久久久久久亚洲| 中文日韩电影网站| 91久久在线视频| 欧美激情综合亚洲一二区| 夜色77av精品影院| 精品国产一区二区在线| 国产福利精品视频| 国产精品久久一区主播| 亚洲欧洲成视频免费观看| 色与欲影视天天看综合网| 亚洲一区二区三区毛片| 成人日韩在线电影| 国产精品欧美激情在线播放| 久久97久久97精品免视看| 欧亚精品在线观看| 成人黄色在线播放| 国产精品久久久久久久久久尿| 国产精品久久久久久久久久免费| 精品久久久999| 国内精品久久久久久| 91免费看片在线| 国语自产精品视频在线看抢先版图片| 91亚洲一区精品| 亚洲男人的天堂在线| 17婷婷久久www| 欧美激情图片区| 欧美高清不卡在线| 欧美大片免费观看在线观看网站推荐| 91久久久久久久久| 色综合影院在线| 亚洲成人激情在线观看| 亚洲国产精品成人精品| 欧美激情第6页| 国产精品久久久久久久久久小说| 91社区国产高清| 超碰精品一区二区三区乱码| 欧美亚洲一级片| 亚洲成人av资源网| 亚洲综合一区二区不卡| 欧洲美女7788成人免费视频| 精品亚洲男同gayvideo网站| 国产69精品久久久久9999| 亚洲美女视频网| 亚洲国产欧美精品| 一夜七次郎国产精品亚洲| 日韩精品久久久久久久玫瑰园| 欧美第一黄色网| 综合欧美国产视频二区| 欧美激情欧美激情在线五月| 国产美女精品视频免费观看| 欧美俄罗斯乱妇| 国产精品久久99久久| 欧美极品少妇xxxxⅹ免费视频| 国产成人午夜视频网址| 欧美高清视频在线播放| 精品亚洲一区二区三区四区五区|