winform 控件隨頁面大小進行自適應


改進C# WinForm窗體及其控件自適應各種屏幕分辨率

一。說明
  2012-11-30 曾經寫過 《C# WinForm窗體及其控件自適應各種屏幕分辨率》 ,其中也講解了控件自適應的原理。近期有網友說,裝在panel里面的控件,沒有效果?

  這是控件嵌套的問題,加入即可實現。使用皮膚時,要注意在窗體Load事件中,有些控件(比如DataGridView)的子控件還沒有完成,在這里記錄控件的個數比較少,而在窗體SizeChanged事件中獲取的比較多(是正常的)。
  解決的方法是,記錄控件的初始位置和大小以及處理控件的縮放,都放到窗體SizeChanged事件中,只需第一次記錄控件的初始位置和大小,之后再調用類的自適應方法即可。
 
 
二。使用方法
  1.把自適應的類整體復制到你的工程命名空間里,(這樣做,每個窗體均可使用)
     然后在需要自適應的窗體中做2步即可:
  2.聲明自適應類實例。
  3.為窗體添加SizeChanged事件,並在其方法Form1_SizeChanged中,首次記錄窗體和其控件初始位置和大小,之后調用類的自適應方法,完成自適應。

三。完整代碼如下:
 
(一)。自適應窗體的代碼:

using System;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    public partial class Form1 : Form
    {
       //1.聲明自適應類實例
       AutoSizeFormClass asc = new AutoSizeFormClass();
       public Form1()
       {
           InitializeComponent();
           //如果加入"皮膚",則不能在Form1_Load中記錄控件的大小和位置,因為有些控件如dataGridView的子控件還未完成
           //而要在在Form1_SizeChanged中,第一次改變時,記錄控件的大小和位置
             this.skinEngine1.SkinFile = "EmeraldColor1.ssk";
       }
       //2. 為窗體添加Load事件,並在其方法Form1_Load中,調用類的初始化方法,記錄窗體和其控件的初始位置和大小
       private void Form1_Load(object sender, EventArgs e)
       {
          // asc.controllInitializeSize(this);
        }
       //3.為窗體添加SizeChanged事件,並在其方法Form1_SizeChanged中,調用類的自適應方法,完成自適應
       private void Form1_SizeChanged(object sender, EventArgs e)
       {
           asc.controlAutoSize(this);
          //  this.WindowState = (System.Windows.Forms.FormWindowState)(2);//記錄完控件的初始位置和大小后,再最大化
     }


    }

}

(二)。自適應類的代碼

using System.Collections.Generic;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    class AutoSizeFormClass
    {
       //(1).聲明結構,只記錄窗體和其控件的初始位置和大小。
       public struct controlRect
       {
           public int Left;
           public int Top;
           public int Width;
           public int Height;
       }
       //(2).聲明 1個對象
       //注意這里不能使用控件列表記錄 List nCtrl;,因為控件的關聯性,記錄的始終是當前的大小。
 //     public List oldCtrl= new List();//這里將西文的大於小於號都過濾掉了,只能改為中文的,使用中要改回西文
       public List《controlRect》 oldCtrl = new List《controlRect》();
       int ctrlNo = 0;//1;
       //(3). 創建兩個函數
       //(3.1)記錄窗體和其控件的初始位置和大小,
       public void controllInitializeSize(Control mForm)
       {
               controlRect cR;
               cR.Left = mForm.Left; cR.Top = mForm.Top; cR.Width = mForm.Width; cR.Height = mForm.Height;
               oldCtrl.Add(cR);//第一個為"窗體本身",只加入一次即可

               AddControl(mForm);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用

            //this.WindowState = (System.Windows.Forms.FormWindowState)(2);//記錄完控件的初始位置和大小后,再最大化
           //0 - Normalize , 1 - Minimize,2- Maximize
       }

       private void AddControl(Control ctl)
       {
           foreach (Control c in ctl.Controls)
           {  //**放在這里,是先記錄控件的子控件,后記錄控件本身
             //if (c.Controls.Count > 0)
             //   AddControl(c);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用
               controlRect objCtrl;
               objCtrl.Left = c.Left; objCtrl.Top = c.Top; objCtrl.Width = c.Width; objCtrl.Height = c.Height;
               oldCtrl.Add(objCtrl);
               //**放在這里,是先記錄控件本身,后記錄控件的子控件
               if (c.Controls.Count > 0)
                   AddControl(c);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用

           }
       }

       //(3.2)控件自適應大小,
       public void controlAutoSize(Control mForm)
       {
           if (ctrlNo == 0)
           { //*如果在窗體的Form1_Load中,記錄控件原始的大小和位置,正常沒有問題,但要加入皮膚就會出現問題,因為有些控件如dataGridView的的子控件還沒有完成,個數少
             //*要在窗體的Form1_SizeChanged中,第一次改變大小時,記錄控件原始的大小和位置,這里所有控件的子控件都已經形成
               controlRect cR;
             //  cR.Left = mForm.Left; cR.Top = mForm.Top; cR.Width = mForm.Width; cR.Height = mForm.Height;
               cR.Left = 0; cR.Top = 0; cR.Width = mForm.PreferredSize.Width; cR.Height = mForm.PreferredSize.Height;

               oldCtrl.Add(cR);//第一個為"窗體本身",只加入一次即可

               AddControl(mForm);//窗體內其余控件可能嵌套其它控件(比如panel),故單獨抽出以便遞歸調用
           }
           float wScale = (float)mForm.Width / (float)oldCtrl[0].Width;//新舊窗體之間的比例,與最早的舊窗體
           float hScale = (float)mForm.Height / (float)oldCtrl[0].Height;//.Height;
           ctrlNo = 1;//進入=1,第0個為窗體本身,窗體內的控件,從序號1開始

           AutoScaleControl(mForm, wScale, hScale);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用
       }

       private void AutoScaleControl(Control ctl, float wScale, float hScale)
       {
           int ctrLeft0, ctrTop0, ctrWidth0, ctrHeight0;
           //int ctrlNo = 1;//第1個是窗體自身的 Left,Top,Width,Height,所以窗體控件從ctrlNo=1開始
           foreach (Control c in ctl.Controls)
           { //**放在這里,是先縮放控件的子控件,后縮放控件本身
                //if (c.Controls.Count > 0)
                //   AutoScaleControl(c, wScale, hScale);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用
               ctrLeft0 = oldCtrl[ctrlNo].Left;
               ctrTop0 = oldCtrl[ctrlNo].Top;
               ctrWidth0 = oldCtrl[ctrlNo].Width;
               ctrHeight0 = oldCtrl[ctrlNo].Height;
               //c.Left = (int)((ctrLeft0 - wLeft0) * wScale) + wLeft1;//新舊控件之間的線性比例
               //c.Top = (int)((ctrTop0 - wTop0) * h) + wTop1;
               c.Left = (int)((ctrLeft0) * wScale);//新舊控件之間的線性比例。控件位置只相對於窗體,所以不能加 + wLeft1
               c.Top = (int)((ctrTop0) * hScale);//
               c.Width = (int)(ctrWidth0 * wScale);//只與最初的大小相關,所以不能與現在的寬度相乘 (int)(c.Width * w);
               c.Height = (int)(ctrHeight0 * hScale);//
               ctrlNo++;//累加序號
               //**放在這里,是先縮放控件本身,后縮放控件的子控件
               if (c.Controls.Count > 0)
                   AutoScaleControl(c, wScale, hScale);//窗體內其余控件還可能嵌套控件(比如panel),要單獨抽出,因為要遞歸調用


                if (ctl is DataGridView)
                {
                    DataGridView dgv = ctl as DataGridView;
                    Cursor.Current = Cursors.WaitCursor;

                    int widths = 0;
                    for (int i = 0; i < dgv.Columns.Count; i++)
                    {
                        dgv.AutoResizeColumn(i, DataGridViewAutoSizeColumnMode.AllCells);  // 自動調整列寬  
                        widths += dgv.Columns[i].Width;   // 計算調整列后單元列的寬度和                       
                    }
                    if (widths >= ctl.Size.Width)  // 如果調整列的寬度大於設定列寬  
                        dgv.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.DisplayedCells;  // 調整列的模式 自動  
                    else
                        dgv.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.Fill;  // 如果小於 則填充  

                    Cursor.Current = Cursors.Default;
                }
           }
       }

    }
}

1。看到幾個用戶提到以下語句錯誤,原因是新浪博客將西文的大於號、小於號中的內容過濾掉了,所以只能改為中文的大於號、小於號,在實際使用中要改回西文

//     public List oldCtrl= new List();//這里將西文的大於小於號都過濾掉了,只能改為中文的,使用中要改回西文
       public List《controlRect》 oldCtrl = new List《controlRect》();
2。在load中正常,到sizechang中,高度和寬度,應使用
mForm.PreferredSize.Width,mForm.PreferredSize.Height;

             //  cR.Left = mForm.Left; cR.Top = mForm.Top; cR.Width = mForm.Width; cR.Height = mForm.Height;
 
              cR.Left = 0; cR.Top = 0; cR.Width = mForm.PreferredSize.Width; cR.Height = mForm.PreferredSize.Height;


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM