C#快速自适应布局

发布时间 2023-08-18 16:48:56作者: 黑夜中的小迷途

一种自适应布局的方法,它可以根据界面比例自动缩放内容控件,在较短时间内完成软件布局。此方法的优点是:简单易学,布局速度快,适用于绝大多数软件界面。缺点是:不太适合需要高度定制的复杂软件界面。实现的原理是:会将from装进Panel里面对控件进行自动计算大小,每次界面变化时,修改每个控件的尺寸。

创建AutoWindowsSize.cs类(复制以下代码即可)

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
//using System.Linq;
using System.Text;
using System.Windows.Forms;

using System.Drawing.Drawing2D;
//using System.Threading.Tasks;


namespace AutoWindowsSize
{
    class AutoAdaptWindowsSize                       //窗口缩放对象
    {
        double formOriginalWidth;//窗体高度原始宽度
        double formOriginalHeight;//窗体原始
        double scaleX;//水平缩放比例
        double scaleY;//垂直缩放比例
        Dictionary<string, string> ControlsInfo = new Dictionary<string, string>();//控件中心Left,Top,控件Width,控件Height,控件字体Size

        private Form _form;
        Panel Win_Panel1 = new Panel();
        public AutoAdaptWindowsSize(Form form)
        {
            _form = form;

            //代码生成一个容器panel1,添加至窗体
            _form.Controls.Add(Win_Panel1);
            Win_Panel1.BorderStyle = BorderStyle.None;    //容器border样式
            Win_Panel1.Dock = DockStyle.Fill;             //设置填充,下面添加控件至容器完成后,容器会填充窗口
            Win_Panel1.BackColor = Color.Transparent;    // 这里默认的背景颜色是form的背景颜色,如果form页面时图片,需要将这里的颜色设置成透明,否则会被覆盖。
            //将窗体所有控件添加至panel1
            while (_form.Controls[0].Name.Trim() != "")
            {
                foreach (Control item in _form.Controls)
                {

                    if (item.Name.Trim() != "" && item.Name.Trim() != Win_Panel1.Name.Trim())
                    {
                        Win_Panel1.Controls.Add(item);
                    }
                }
            }

            //保存窗体和控件初始大小
            InitControlsInfo(Win_Panel1);
        }

        public void InitControlsInfo(Control ctrlContainer)
        {
            if (ctrlContainer.Parent == _form)//获取窗体的高度和宽度
            {
                formOriginalWidth = Convert.ToDouble(ctrlContainer.Width);
                formOriginalHeight = Convert.ToDouble(ctrlContainer.Height);
            }
            foreach (Control item in ctrlContainer.Controls)
            {
                if (item.Name.Trim() != "")
                {
                    //添加信息:键值:控件名,内容:据左边距离,距顶部距离,控件宽度,控件高度,控件字体。
                    ControlsInfo.Add(item.Name, (item.Left + item.Width / 2) + "," + (item.Top + item.Height / 2) + "," + item.Width + "," + item.Height + "," + item.Font.Size);
                }
                if ((item as UserControl) == null && item.Controls.Count > 0)
                {
                    InitControlsInfo(item);
                }
            }

        }
        public void FormSizeChanged()
        {
            try
            {
                if (ControlsInfo.Count > 0)//如果字典中有数据,即窗体改变
                {
                    ControlsZoomScale(Win_Panel1);//表示pannel控件
                    ControlsChange(Win_Panel1);
                }
            }
            catch { }
        }
        private void ControlsZoomScale(Control ctrlContainer)
        {
            scaleX = (Convert.ToDouble(ctrlContainer.Width) / formOriginalWidth);
            scaleY = (Convert.ToDouble(ctrlContainer.Height) / formOriginalHeight);
        }

        // 改变控件大小

        private void ControlsChange(Control ctrlContainer)
        {
            double[] pos = new double[5];//pos数组保存当前控件中心Left,Top,控件Width,控件Height,控件字体Size
            foreach (Control item in ctrlContainer.Controls)//遍历控件
            {
                if (item.Name.Trim() != "")//如果控件名不是空,则执行
                {
                    if ((item as UserControl) == null && item.Controls.Count > 0)//如果不是自定义控件
                    {
                        ControlsChange(item);//循环执行
                    }
                    string[] strs = ControlsInfo[item.Name].Split(',');//从字典中查出的数据,以‘,’分割成字符串组

                    for (int i = 0; i < 5; i++)
                    {
                        pos[i] = Convert.ToDouble(strs[i]);//添加到临时数组
                    }
                    double itemWidth = pos[2] * scaleX;     //计算控件宽度,double类型
                    double itemHeight = pos[3] * scaleY;    //计算控件高度
                    item.Left = Convert.ToInt32(pos[0] * scaleX - itemWidth / 2);//计算控件距离左边距离
                    item.Top = Convert.ToInt32(pos[1] * scaleY - itemHeight / 2);//计算控件距离顶部距离
                    item.Width = Convert.ToInt32(itemWidth);//控件宽度,int类型
                    item.Height = Convert.ToInt32(itemHeight);//控件高度
                    if (float.Parse((pos[4] * Math.Min(scaleX, scaleY)).ToString()) != 0)         //缩放字体大小不能为0
                    { item.Font = new Font(item.Font.Name, float.Parse((pos[4] * Math.Min(scaleX, scaleY)).ToString())); }  //字体
                }
            }
        }
    }
}

2.创建结束以后,在Form1引入using AutoWindowsSize;

using AutoWindowsSize;

3.在Form1中设置全局变量AutoSize。

AutoAdaptWindowsSize AutoSize;

4.在Form1_Load添加如下代码,在界面初始化的过程中创建AutoSize。

private void Form1_Load(object sender, EventArgs e)
{
    AutoSize = new AutoAdaptWindowsSize(this);
}

5.创建Form1_SizeChanged事件函数,一定加这个AutoSize != null判断,电脑缩放布局不是100%的时候,会报错。代码如下:

private void Form1_SizeChanged(object sender, EventArgs e)

{
    if (AutoSize != null) // 一定加这个判断,电脑缩放布局不是100%的时候,会报错
    {
        AutoSize.FormSizeChanged();
    }
}

以上部分就实现了,自适应布局的代码步骤。但在使用过程中,可能会有以下问题,可更具需要进行修改:

1.Form页面如果是背景图片,页面更新特别快的时候(比如有图片更新),页面会有闪烁情况,如果纯色不会有此类情况,暂时没有解决。

2.如果Form中的背景是图片,会直接显示成纯图的背景,原因是在AutoAdaptWindowsSize类中会创建一个panel,panel背景颜色会自动设置,而pannel的纯色背景会覆盖掉Form1的背景,本质上是panel的背景。所以可以给Panel设置背景图片或将panel背景色修改为Transparent,如下:

Win_Panel1.BackColor = Color.Transparent;

我已经将类里面的内容做了修改,可以直接忽略。

3.其次自适应布局,每次修改窗口需要对界面重新绘制,在拖动窗口时会出现比较严重的闪屏现象,需要在Form1的Disiger中添加任意位置添加如下代码:

// 引入Forms。
using System.Windows.Forms;
protected override CreateParams CreateParams
{
    get
    {
        CreateParams cp = base.CreateParams;
        cp.ExStyle |= 0x02000000;
        return cp;
    }
}

以上就是自适应布局的内容,经过测试可以满足我们绝大部分的界面自适应需求。
引用出处:作者:666号特派员 https://www.bilibili.com/read/cv23844020 出处:bilibili