一。說明
2012-11-30 曾經寫過 《C# WinForm窗體及其控件自適應各類屏幕分辨率》 ,其中也講解了控件自適應的原理。近期有網友說,裝在panel裏面的控件,沒有效果?html
這是控件嵌套的問題,加入便可實現。使用皮膚時,要注意在窗體Load事件中,有些控件(好比DataGridView)的子控件尚未完成,在這裏記錄控件的個數比較少,而在窗體SizeChanged事件中獲取的比較多(是正常的)。
解決的方法是,記錄控件的初始位置和大小以及處理控件的縮放,都放到窗體SizeChanged事件中,只需第一次記錄控件的初始位置和大小,以後再調用類的自適應方法便可。
二。使用方法
1.把自適應的類總體複製到你的工程命名空間裏,(這樣作,每一個窗體都可使用)
而後在須要自適應的窗體中作2步便可:
2.聲明自適應類實例。
3.爲窗體添加SizeChanged事件,並在其方法Form1_SizeChanged中,首次記錄窗體和其控件初始位置和大小,以後調用類的自適應方法,完成自適應。
三。完整代碼以下:
(一)。自適應窗體的代碼:函數
using System;
using System.Windows.Forms;this
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);//記錄完控件的初始位置和大小後,再最大化
}spa
}orm
}htm
(二)。自適應類的代碼對象
using System.Collections.Generic;
using System.Windows.Forms;blog
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),要單獨抽出,由於要遞歸調用
}
}
}
}
20140627特此更正:
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;