C# ——窗體和控件隨着分辨率的變化自適應大小


一.說明

  我們自己編寫程序的界面,會遇到各種屏幕分辨 率,只有自適應才能顯的美觀。實際上,做到這點也很簡單,就是首先記錄窗體和它上面控件的初始位置和大小,當窗體改變比例時,其控件的位置和大小也按此比 例變化即可。因為窗體上控件的位置和大小是相對於自己所在的窗體的,也就是所謂的窗口坐標。
  在這里我們只考慮相對於自己窗體的窗口坐標更簡單,也就是成比例變化。為了多個窗體共用,我在這里創建一個類AutoSizeFormClass  ,1.使用它去記錄窗體和其控件的初始位置和大小,2.根據窗體變化了的大小,成比例地實現其控件的水平和垂直方向的變化,也就是自適應。
 
二.使用方法
  使用方法很簡單,
  1.把自適應的類整體復制到你的工程命名空間里,
     然后在需要自適應的窗體中做3步即可:
  2.聲明自適應類實例。
  3.為窗體添加Load事件,並在其方法Form1_Load中,調用類的初始化方法,記錄窗體和其控件初始位置和大小
  4.為窗體添加SizeChanged事件,並在其方法Form1_SizeChanged中,調用類的自適應方法,完成自適應
 
三.完整代碼如下:
 
 
1.定義AutoResizeForm類:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace CSharpFormApplication
{
    class AutoResizeForm
    {
            //(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;
                    }
                }


            }
        }
    }

2.在要自適應大小的窗體中聲明全局類對象

//1.聲明自適應類實例
        AutoResizeForm asc = new AutoResizeForm();

3.在窗體的Load事件中調用類的初始化方法:

 //2.位窗體添加Load時間,並在其中調用類的初始化方法,記錄窗體和其控件的初始位置和大小
        private void AutoResizeWithResolutionForm_Load(object sender, EventArgs e)
        {
            asc.controllInitializeSize(this);
        }

4.在窗體的SizeChange事件中調用窗體自適應方法:

        //3.為窗體添加SizeChanged事件,並在其方法Form1_SizeChanged中,調用類的自適應方法,完成自適應
        private void AutoResizeWithResolutionForm_SizeChanged(object sender, EventArgs e)
        {
            asc.controlAutoSize(this);
        }

5.實現效果:

 

 

當然,窗口坐標和屏幕坐標也是可以相互轉換的,  

 

private void Form1_MouseDown(object sender, MouseEventArgs e)
{
   int x = e.X; //相對form窗口的坐標,客戶區坐標
   int y = e.Y;
   int x1 = Control.MousePosition.X;//相對顯示器,屏幕的坐標
   int y1 = Control.MousePosition.Y;  
}


它們之間轉換如下:
this.Location; // 窗體所在坐標
this.PointToScreen(new Point(0, 0)); // 客戶區坐標轉換為屏幕坐標
this.PointToClient(new Point(0, 0)); // 屏幕坐標轉換為客戶區坐標


免責聲明!

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



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