元件製作之四(外觀設計)

時常想,如果一個元件能夠按自己想要的外觀顯示,那該是件多麼COOL的事啊,這一篇就要來做一個精美外觀的元件,但是,做什麼好呢.Button? < SPAN>高手突破>有關於自己定義外觀的Button,以及CheckBox等的做法,Button從CustomPanel繼承,重載 Paint方法來畫外觀.如果你有興趣,可以去找來看,這裏就不做Button了,做一個Memo如何呢.?是個不錯的主意。
 
我們先起個名字叫做TCoolMemo。以上篇已經講了很多組件的技術,這裏就只說出幾個重點。其餘不多說了。
 
首先,該Memo從CustomMemo繼承,它有這樣外觀:屬於平面的,邊框是可以設置顏色的線,對應的顏色變數為FEdgeColor,另外,離邊框 以內的兩個象素處,還有另一個框,當滑鼠進入Memo時,這個框會顯示,當滑鼠離開時,為個框消失,同樣也可以設置顏色,對應變數為 FEnterColor。
那麼滑鼠進入和離開怎麼判斷呢,這裏Memo將截獲兩個Delphi的內部消息:
//下面兩個獲得Delphi的內部消息,滑鼠進入和離開時發生
     procedure CMMouseEnter (var Message: TMessage); message CM_MOUSEENTER;
     procedure CMMouseLeave (var Message: TMessage); message CM_MOUSELEAVE;
其實父類已經截獲了這兩個消息,並作了相應處理,所以TCoolMemo中的消息處理函數要
Inherited;再作自己的處理。這裏又用到了一個變數
MouseIn:Boolean;//標識滑鼠是否進入元件
 
接下來TCoolMemo還要截獲兩個消息:
procedure WMPaint (var Message: TMessage); message WM_PAINT;
procedure WMNCCalcSize (var Message: TWMNCCalcSize); message WM_NCCALCSIZE;
第一個很熟悉,當需要重畫時,觸發該消息,
第二個是當表單需要計算位置和尺寸時觸發,消息中包含了視窗客戶區的大小,我們用這個的目的主要是將客戶區縮小三個象素,以便畫元件時不會畫到客戶區。
procedure TCoolMemo.WMNCCalcSize (var Message: TWMNCCalcSize);
begin
  inherited;
  InflateRect(Message.CalcSize_Params^.rgrc[0], -3, -3);
end;
 
而上面幾個消息處理函數,CM_MOUSEENTER和CM_MOUSELEAVE;將引起TCoolMemo的外觀變化,WM_PAINT保存其外觀不被擦去。所以要用到一個畫元件的函數,即:
drawBorder;
裏面用到了幾個API的GDI函數。我在代碼中有詳細的說明,加上自己看幫助,應該是可以看懂的。
 
另外,相比於Memo,它的擴展了這樣的功能:設置邊距和獲得游標的位置。這兩個對應的性屬為Margin,Position。他們都是Public的,不可以在物件察看器中看到。
我們一個個來說
邊距設置
property Margin:byte read FMargin write setMargin default 0;
其中setMargin函數中發送了兩個消息:
//該消息取得輸入區的尺寸
SendMessage(Handle, EM_GETRECT, 0, Longint(@Rect));
//該消息設定輸入區的大小
SendMessage(Handle, EM_SETRECT, 0, Longint(@Rect));
 
游標的位置:
property Position:TPosition read getPosition;
TPostion是一個結構,其中有行和列兩個值:
TPosition=record  //指定游標的行和列
     row:longint;
     col:longint;
   end;
getPosition;中還要處理中文的問題,代碼有詳細說明,如果文本中有中文,一樣也可以得到正確的行和列。
 
最後增加了兩個事件
property OnEnter;
property OnExit;
都是從父類中顯化出來的,其實就是CM_MOUSEENTER和CM_MOUSELEAVE;消息引起的。,當你想作一個三態按鈕,這兩個事件很有作用。
 
好了,重點就是上面那幾個了,以下是源代碼,其中也有詳細的說明:
 
unit CoolMemo;
 
interface
 
uses
   Windows, Messages, Classes, Forms,Controls, Graphics, StdCtrls;
 
type
    //用設定邊緣的空白
   TPosition=record  //指定游標的行和列
     row:longint;
     col:longint;
   end;
   TCoolMemo=class(TCustomMemo)
   private
     FMargin:byte;  //邊距的大小
     FEdgeColor:TColor;//邊框的顏色
     FEnterColor:TColor;//滑鼠進入時邊框內側的框顏色
     MouseIn: Boolean; //標識滑鼠是否進入
     function getPosition:TPosition;//游標的行和列
     procedure setMargin(value:byte);
     procedure setEdgeColor(Value:TColor);
     procedure setEnterColor(Value:TColor);
     //下面兩個獲得Delphi的內部消息,滑鼠進入和離開時發生
     procedure CMMouseEnter (var Message: TMessage); message CM_MOUSEENTER;
     procedure CMMouseLeave (var Message: TMessage); message CM_MOUSELEAVE;
     //當一個視窗的外觀必須被畫時,應用程式發送這個消息給該視窗
     procedure WMPaint (var Message: TMessage); message WM_PAINT;
     //表單需要計算位置和尺寸時觸發
     //我們用這個的目的主要是將客戶區縮小三個象素,以便畫元件時不會畫到客戶區。
  procedure WMNCCalcSize (var Message: TWMNCCalcSize); message WM_NCCALCSIZE;
   protected
   //畫表單的邊框,使其看起來更美觀.
     procedure drawBorder;
   public
     constructor Create (AOwner: TComponent); override;
     property Position:TPosition read getPosition;
     property Margin:byte read FMargin write setMargin default 0;
   published
    property EdgeColor:TColor read FEdgeColor write SetEdgeColor default $ff0000;
    property EnterColor:TColor read FEnterColor write SetEnterColor default $0000ff;
    //顯式化父類的屬性
    property Align;
    property Alignment;
    property DragCursor;
    property DragMode;
    property Enabled;
    property Color;
    property Font;
    property Lines;
    property MaxLength;
    property OEMConvert;
    property ParentFont;
    property ParentShowHint;
    property PopupMenu;
    property ReadOnly;
    property ShowHint;
    property ScrollBars;
    property TabOrder;
    property TabStop;
    property Visible;
    property WantReturns;
    property WantTabs;
    property WordWrap;
 
    property OnChange;
    property OnClick;
    property OnDblClick;
    property OnDragDrop;
    property OnDragOver;
    property OnEndDrag;
    //增加這兩個事件,處理滑鼠進入和離開
    property OnEnter;
    property OnExit;
    property OnKeyDown;
    property OnKeyPress;
    property OnKeyUp;
    property OnMouseDown;
    property OnMouseMove;
    property OnMouseUp;
    property OnStartDrag;
   end;
 
procedure Register;
 
implementation
 
procedure Register;
begin
  RegisterComponents('Samples', [TCoolMemo]);
end;
 
constructor TCoolMemo.Create(AOwner:TComponent);
begin
  inherited Create(Aowner);
  ControlStyle := ControlStyle - [csFramed];
  ParentFont := True;
  FEdgeColor := $ff0000;
  FEnterColor := $0000ff;
  //設定外觀,平面無邊形
  Ctl3D := False;
  FMargin:=0;
  BorderStyle:=bsNone;
  height:=150;
  width:=200;
end;
 
procedure TCoolMemo.setMargin(Value:byte);
var
  Rect: TRect;
begin
//該消息取得客戶區的尺寸
  SendMessage(Handle, EM_GETRECT, 0, Longint(@Rect));
  //以下是重新確定尺寸
  Rect.Top := Value;
  Rect.Left := Value;
  Rect.Right := Width -Value;
  Rect.Bottom := Height -Value;
//該消息設定客戶區的大小
  SendMessage(Handle, EM_SETRECT, 0, Longint(@Rect));
  Fmargin:=value;
end;
 
function TCoolMemo.getPosition:TPosition;
var
  row,Col:longint;
  CBLines:longint;
  str:WideString;
begin
//該消息取得游標所在的行,
  row:= SendMessage(Handle,EM_LINEFROMCHAR,SelStart,0);
  //該消息取得游標所在行開始的位置,位置從第一行的0開始計數,
  //每過一個字元增加1,
  CBLines:=SendMessage(Handle,EM_LINEINDEX,row,0);
  //得到游標的所在行的所在列
  Col:=SelStart-CBLines;
  //為了解決中文的問題,需要用寬字元型來取得游標所在行
  //,行中游標所在列之前的字串,這樣可以解決中文列數的確定問題.
  str:=Copy(Lines[row],1,col);
  col:=Length(Str)+1;
  result.row:=row+1;
  result.col:=col;
end;
 
procedure TCoolMemo.setEdgeColor(Value:TCOlor);
begin
 if FEdgeColor<>value then
 begin
  FEdgeColor:=value;
  drawBorder;
 end;
end;
 
procedure TCoolMemo.setEnterColor(Value:TColor);
begin
 if FEnterColor<>value then
 begin
   FEnterColor:=value;
   drawBorder;
 end;
end;
 
procedure TCoolMemo.CMMouseEnter(var Message: TMessage);
begin
  inherited;
    MouseIn:= True;
    drawBorder;
end;
 
procedure TCoolMemo.CMMouseLeave(var Message:TMessage);
begin
  inherited;
  MouseIn:=False;
  drawBorder;
end;
 
procedure TCoolMemo.WMPaint (var Message: TMessage);
begin
  inherited;
  drawBorder;
end;
 
procedure TCoolMemo.WMNCCalcSize (var Message: TWMNCCalcSize);
begin
  inherited;
  InflateRect(Message.CalcSize_Params^.rgrc[0], -3, -3);
end;
 
procedure TCoolMemo.drawBorder;
var
  DC: HDC;  //設備描述表
  R: TRect; //客戶區
  EnterBrush,OuterBrush,BorderBrush:HBRUSH;  //畫筆控制碼,API
begin
  DC:= GetWindowDC(Handle);  //取得該元件的設備描述表
  try
    GetWindowRect(Handle, R);  //取得該元件的客戶區尺寸
    OffsetRect(R, -R.Left, -R.Top); //左上偏移
    //創建畫筆,兩個,分別代碼邊框,邊框內,白色畫筆
    BorderBrush := CreateSolidBrush(ColorToRGB(FEdgeColor));
    EnterBrush:= CreateSolidBrush(ColorToRGB(FEnterColor));
    OuterBrush:=CreateSolidBrush(ColorToRGB(clWhite));
//not(csDesigning in ComponentState保證在設計期不變
    if (not(csDesigning in ComponentState)) and
    (MouseIn=true) then  //如果滑鼠進入
    begin
      //畫一個矩形框,用BorderBrush畫筆
      FrameRect(DC, R, BorderBrush);
      //把R縮小一個象素
      InflateRect(R, -1, -1);
      //畫一個矩形框,用outerBrush畫筆
      FrameRect(DC, R, outerBrush);
      InflateRect(R, -1, -1);
      FrameRect(DC, R, EnterBrush);
    end
    else  //如果滑鼠沒有進入
    begin
      FrameRect(DC, R, BorderBrush);
      InflateRect(R, -1, -1);
      FrameRect(DC, R, outerBrush);
      InflateRect(R, -1, -1);
      FrameRect(DC, R, outerBrush);
    end;
  finally
    ReleaseDC(Handle, DC);  //釋放設備描述表
  end;
  DeleteObject(BorderBrush);   //釋放畫筆
  DeleteObject(EnterBrush);
  DeleteObject(OuterBrush);
end;
 
end.
 
安裝上去試試吧,比Memo1好看多了,功能也強多了。是嗎。
 
至此已經做了三個組件了,其實不算很複雜,只要理清思緒。到這裏似乎可以結束這次的元件製作之旅了,但是還沒有。我們似乎還沒有做過非視覺化組件。所以我想最後一個,就是做一個非視覺化元件。想知道是什麼,往下看吧。

posted on 2005-09-07 10:16  Jason Cheng  阅读(698)  评论(0编辑  收藏  举报