程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> 關於C++ >> 深入C++ Builder之編寫自己的元件-深入分析VCL繼承、消息機制(1)

深入C++ Builder之編寫自己的元件-深入分析VCL繼承、消息機制(1)

編輯:關於C++

這篇文章提及內容可能大家已經在很多地方看到過了,作者也是如此,只不過還看了很多VCL源代碼,加上自己實際編寫元件的經驗,拼湊了這麼一篇文章。所以所有言論都是個人觀點、經驗的描述,僅供參考。

你可轉載,拷貝,但必須加入作者署名Aweay,如果用於商業目的,必須經過作者同意。

系統要求

如果你想一起跟著做的話,那麼你應該看看這裡,否則你可以直接跳過。

C++ Builder6 + updata4 (上帝造人的工具,以下簡稱BCB)

Windows2k or higher (必要)

作者強烈建議你使用WinNT,BCB在Win9x下有非常多的問題,而且非常不穩定,就算你不在乎這個,還有一個非常致命的問題,BCB的幫助文件在Win9x下顯示不完全(因為BCB的幫助索引關鍵字數量超過Win9x的限制),這樣非常難於參考幫助。

Delphi6 :( (必要)

什麼?是不是寫錯了,完全沒有寫錯,如果你要深入VCL查看源代碼的話,在沒有比用Delphi6更合適的了,在全部安裝Delphi6後,把VCL Source的目錄加入Search Path中,這樣你可以在編輯器中按住Ctrl鍵,點擊鼠標直接跳轉到源代碼處非常方便,比什麼grep好用多了。

起步

對於VCL的消息機制,大家可以參考CKER的

http://www.csdn.net/develop/read_article.asp?id=8131

重復的內容我就不介紹了,但是對於編寫元件來說上面的消息機制還是很模糊,而且很多時候並不是用那些方法來處理消息的,還有就是元件特有的CM_XXXXXXXX消息如何處理呢?如何加入自己的事件呢?這些問題我會在後面的討論中做詳細介紹。

站在巨人的肩膀

編寫元件的第一件事情就是確定我們從那裡繼承的問題,選取一個好的祖先類是編寫一個好的元件的第一步,那麼到底如何選取他山之石呢?一般性的規則是這樣的:

1.對於有界面的顯示的,需要處理鍵盤事件的,又不是容器的組件從TCustomControl繼承

2.對於有界面的顯示的,需要不處理鍵盤事件的,需要處理鼠標事件的從TGraphicsControl繼承

3.對於沒有界面顯示的,類似與TOpenDialog/TXpMenu這樣的控件從TComponent繼承

4.如果你想擴展某個指定的控件,比如TPanel,你最好從TCustomPanel繼承,而不要從TPanel直接繼承。

注意上面第4條規則,基本上所有組件都有TCustomXXX的父類,這也是VCL鼓勵的繼承對象,原因在於你可以定制元件屬性的可見性,最重要的是他們的構造函數和析構函數是虛擬的。

這篇文章主要針對1,2規則的元件進行介紹,3,4相對簡單就不作深入討論了。

畫出自己

元件要顯示在窗體上,必須以一定的樣子出現,那麼可定要畫出自己,大家都知道處理WM_PAINT消息就可以了,從CKER的文章裡,我們可以得出很多方法來處理這個消息,比如:

__fastcall WndProc(TMessage msg)
{
  switch(msg->msg)
  {
   case WM_PAINT:
    //我們的處理代碼
  ...
}

或者干脆用消息映射的宏,但這些都不是最好的方法。

從TControl以後的組件都有Paint這個虛擬方法,我們只要重載這個方法就可以自動繪制,相當於處理了WM_PAINT,這是因為:

procedure TGraphicControl.WMPaint(var Message: TWMPaint);
begin
  if Message.DC <> 0 then
  begin
   Canvas.Lock;
   try
    Canvas.Handle := Message.DC;
    try
     Paint;
    finally
     Canvas.Handle := 0;
    end;
   finally
    Canvas.Unlock;
   end;
  end;
end;

以上代碼片斷說明了這一點,據我所研究過的專業級組件都是通過重載這個函數來繪制自己的。

注意上面的代碼片斷就是用我上面提到的方法(裝delphi6)按了幾次鼠標左鍵得到的,是不是很實惠,^_^。

在Paint方裡我們可以自由繪制,在後面的文章裡我會交大家如何高效率繪制。

在很多時候,我們需要重繪自己,比如我前幾天給網友做的劃線的組件,當線的寬度改變時我們必須重繪自己,否則無法反映屬性的改變,我見很多朋友使用repaint()方法,這也不是最好的方法,我們應該用Invalidate(),為什麼?留給大家看源代碼吧,就算復習上面的知識了。

代碼演示:

void __fastcall TLine::SetLineWidth(int value)
{
  //TODO: Add your source code here
  if(FLineWidth!=value)
  {
   FLineWidth=value;
   Invalidate();
  }
}

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved