您的位置:首页 > 编程语言 > C#

C#winForm窗体自适应改进

2017-05-31 14:42 337 查看

C#WinForm窗体自适应改进

WinForm窗体程序编程相对于WPF和C++编写的MFC、QT简单好写,但微软却没有为自适应填坑。

感觉一般解决自适应有三个解决办法:

1.像安卓编写用XML文件为每个分辨率的电脑进行配置,包括字体、控件位置等。

2.通过格局控件去为页面排好版(但字体不能满足自适应)

3.通过递归去改变每个控件的属性(消耗大,可能会出现闪烁)

以下是我参考:

http://www.cnblogs.com/gguozhenqian/p/4288451.html

1.自适应类:

class AutoSizeFormClass
{
//(1).声明结构,只记录窗体和其控件的初始位置和大小。
public struct controlRect
{
public int Left;
public int Top;
public int Width;
public int Height;
public float SizeFont;
}

//(2).声明 1个对象
//注意这里不能使用控件列表记录 List nCtrl;,因为控件的关联性,记录的始终是当前的大小。
//这里将西文的大于小于号都过滤掉了,只能改为中文的,使用中要改回西文
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; cR.SizeFont = mForm.Font.Size;
oldCtrl.Add(cR);//第一个为"窗体本身",只加入一次即可
AddControl(mForm);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
}

/// <summary>
/// 递归添加控件
/// </summary>
/// <param name="ctl">控件</param>
private void AddControl(Control ctl)
{
foreach (Control c in ctl.Controls)
{ //**放在这里,是先记录控件的子控件,后记录控件本身
controlRect objCtrl;
objCtrl.Left = c.Left; objCtrl.Top = c.Top; objCtrl.Width = c.Width; objCtrl.Height = c.Height; objCtrl.SizeFont = c.Font.Size;
oldCtrl.Add(objCtrl);
//**放在这里,是先记录控件本身,后记录控件的子控件
if (c.Controls.Count > 0)
{
AddControl(c);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
}
}
}

/// <summary>
/// (3.2)控件自适应大小,
/// </summary>
/// <param name="mForm">主窗体</param>
public void controlAutoSize(Control mForm)
{
if (ctrlNo == 0)
{ //*如果在窗体的Form1_Load中,记录控件原始的大小和位置,正常没有问题,但要加入皮肤就会出现问题,因为有些控件如dataGridView的的子控件还没有完成,个数少
//*要在窗体的Form1_SizeChanged中,第一次改变大小时,记录控件原始的大小和位置,这里所有控件的子控件都已经形成
controlRect cR;
cR.Left = 0; cR.Top = 0; cR.Width = mForm.Width; cR.Height = mForm.Height; cR.SizeFont = mForm.Font.Size;
//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),要单独抽出,因为要递归调用
}

/// <summary>
/// 改变控件位置、大小、字体大小
/// </summary>
/// <param name="ctl"></param>
/// <param name="wScale">新旧窗体宽度的比例</param>
/// <param name="hScale">新旧窗体的比例</param>
private void AutoScaleControl(Control ctl, float wScale, float hScale)
{

float sizeFont = wScale > hScale ? wScale : hScale;//字体变化倍数
int ctrLeft0, ctrTop0, ctrWidth0, ctrHeight0;
float ctrSizeFont0;
foreach (Control c in ctl.Controls)
{
//**放在这里,是先缩放控件的子控件,后缩放控件本身
ctrLeft0 = oldCtrl[ctrlNo].Left;
ctrTop0 = oldCtrl[ctrlNo].Top;
ctrWidth0 = oldCtrl[ctrlNo].Width;
ctrHeight0 = oldCtrl[ctrlNo].Height;
ctrSizeFont0 = oldCtrl[ctrlNo].SizeFont;

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);
c.Font = new Font(c.Font.FontFamily,ctrSizeFont0 * sizeFont);
ctrlNo++;//累加序号
//**放在这里,是先缩放控件本身,后缩放控件的子控件
if (c.Controls.Count > 0)
{
AutoScaleControl(c, wScale, hScale);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
}

//不同控件自适应修改
if (ctl.GetType() == typeof(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).声明自适应类实例  
AutoSizeFormClass asc = new AutoSizeFormClass();
 (2)双缓冲:缓解页面重绘时闪屏问题

//设置双缓冲解决控件发生改变闪烁问题
SetStyle(ControlStyles.UserPaint, true);
SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景.
SetStyle(ControlStyles.DoubleBuffer, true); // 双缓冲
 (3)应用

/// <summary>
/// RegisterFrm大小改变时触发
/// </summary>
private void RegisterFrm_SizeChanged(object sender, EventArgs e)
{
//调用类的自适应方法,完成自适应
asc.controlAutoSize(this);
}本案例可以解决
1.控件适应

2.文字适应

3.如果有其他特殊控件或应用皮肤可以向自适应类中添加代码
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: