首頁  >  文章  >  後端開發  >  C# GDI+程式設計(五)

C# GDI+程式設計(五)

高洛峰
高洛峰原創
2016-12-17 10:09:471405瀏覽

呼叫API函數,在視窗非客戶區繪圖

GDI+的Graphics類別裡有個FromHdc函數,這個函數可以根據視窗裝置上下文(DC)建立Graphics對象,在vc++中,視窗客戶區與非客戶區的繪圖無非就是GetWindowDC和GetDC函數的不同呼叫。前者獲得整個視窗DC,後者獲得窗口客戶區DC。

那麼我們就可以在C#裡,呼叫GetWindowDC函數取得整個視窗DC,然後透過FromHdc載入進去,這樣我們就能針對整個視窗繪圖了。

C#要如何呼叫WINDOWS API呢,或是說如何呼叫動態連結函式庫(DLL)裡的函式。

跟VC++的大同小異,先導入動態連結函式庫,然後再宣告API函數,如下:

 [System.Runtime.InteropServices.DllImport("User32.dll")]IntDC private static

當然上面是最簡單的,還有一些細節沒講,先就這樣吧,會基本使用就行了,那些細節問題以後再詳細說明。

在C#中,我們發現API函數的參數型別都不一樣了,像是在VC++中的句柄HDC,HWND。在這裡聲明時,都用了IntPtr代替,這是沒有辦法的事,因為C#沒有指標這個概念,而我們透過查HDC,和HWND類型定義時發現,它們都是指標類型。

所以在C#中,這些「句柄」類型都用IntPtr代替,包括區域句柄HRGN,HICON圖標,HFONT字體句柄等。

看一個範例吧,(接著上一章的)

    public partial class Form1 : Form
    {
        //導入動態連結函式庫,且宣告函數,此函數是宣告在Form1類別中的宣告函數。
        [System.Runtime.InteropServices.DllImport("User32.dll")]
         //儲存PNG非透明部分的路徑
        private GraphicsPath path = new GraphicsPath();
        / /載入PNG圖片
        Bitmap bmp = new Bitmap("d:\Image\win.png");
             InitializeComponent();
            //判斷每像素的色彩值,以取得圖片的顯示區域
            for (int y = 0; y     bmp.Width; x++)
                {
         
                    int argb = cor.ToArgb();
                    byte[] bargb = BitConverter.GetBytes(argb);
                    //像素顏色值不是透明的
                    if (bargb[3] != 0)
                    {
                        //把這個像素點區域加入到路徑裡去
                        path.AddRectangle(new        }
                }
         ion = new Region(path) ;
            this.Paint += formPaint;
         
   sender, PaintEventArgs e)
        {
            
     //Handle是視窗句柄,它是IntPtr類型
            IntPtr hdc = GetWindowDC (this.Handle);
            //依照視窗DC建立Graphics物件
            /繪製圖片
            gr.DrawImage(bmp, new Rectangle(0, 0, bmp.Width, bmp .Height));
        }
protected override void OnPaintBackground(PaintEventArgs e)
        {
          ground(e);
            e.Graphics.FillRectangle(Brushes.Transparent, this.ClientRectangle);
   怎麼樣,效果不錯吧,但一拖曳窗口就原形畢露了,注意到蘋果下方的陰影了麼,就是為了實現這個效果才會帶來一些問題,或者說麻煩了許多吧。只是我沒去解決。移動窗口,或最大化窗口,都沒有完全刷新整個窗口,才會導致這種問題出現。這個問題留待以後解決吧,

在興趣的朋友也可以去解決這個問題。

另外,我用透明畫刷填充的只是視窗的客戶區,如果想填充整個視窗(包括標題列),方法跟在整個視窗繪圖一樣,獲得WindowDC,然後

創建Graphics對象,繪製視窗背景。

(題外話:在vc++中,客戶區與非客戶區有著不同的重繪訊息,WM_PAINT和WM_NCPAINT,這一點要注意了,在刷新非客戶區的時候,別重繪客戶區,雖說不會出什麼問題,但影響了效率總是不好的,能避免就避免)

自繪視窗非客戶區(包括標題欄,最大,最小化,關閉按鈕)

重寫訊息處理函數WndProc

public partial class Form1 : Form

    {

        public Form1()

              }

        protected override void WndProc(ref Message m) == 0xA3)//WM_NCLBUTTONDBLCLK  雙擊標題訊息

                MessageBox.Show("你雙擊了標題欄");
        ref m);
        }

    }

這樣雙擊標題欄的時候就會給予一個提示,然後再預設處理。

查訊息對應的數值,可以到VC++編譯器裡去查,例如打上WM_LBUTTONDOWN然後右鍵,選擇轉到定義就可以查看了。

m.HWnd儲存有視窗句柄,m.LParam和m.WParam是訊息的附帶訊息,可以參考CreateWindow函數裡的WPARAM和LPARAM參數來解釋。

 

 自繪非客戶區工作量實在是太大了,這裡我只給個大概的思路,方向,以後有空再來做吧。

前提當然是把各項資料計算出來,例如視窗有無邊框,如果有的話,取得邊框寬度,高度,然後計算四個邊框的矩形區域。

最後就判斷視窗有無最大,最大小化屬性,然後得到三個按鈕的區域。

而SystemInformation類別裡就儲存有這些數據,例如SystemInformation.CaptionButtonSize儲存有標題欄按鈕的大小,得到了大小,就可以

確定按鈕的區域了,因為這三個按鈕都在視窗的右上角,除去邊框的高寬。

而SystemInformation.CaptionHeight儲存有標題列的高度,邊框的高寬儲存在SystemInformation.BorderSize或SystemInformation.Border3DSize,這個根據視窗的FormBorderStyle決定。視窗的是否處於最大化可以判斷MaximizeBox,為true最大化。

得到了上面那些數據,就響應非客戶區的各種訊息,如滑鼠左鍵訊息WM_NCLBUTTONDOWN和WM_NCLBUTTONUP。

滑鼠移動訊息WM_NCMOUSEMOVE,接著就開始自繪了。

另Rectangle類別裡的Contains函數,可以判斷一個點是否在一個矩形區域內。

 

更多C# GDI+程式(五)相關文章請關注PHP中文網!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
上一篇:淺談C#使用GDI+下一篇:淺談C#使用GDI+